Refactor_Expenses #321
| @ -6,34 +6,27 @@ import RenderAttendanceStatus from "./RenderAttendanceStatus"; | |||||||
| import usePagination from "../../hooks/usePagination"; | import usePagination from "../../hooks/usePagination"; | ||||||
| import { useNavigate } from "react-router-dom"; | import { useNavigate } from "react-router-dom"; | ||||||
| import { ITEMS_PER_PAGE } from "../../utils/constants"; | import { ITEMS_PER_PAGE } from "../../utils/constants"; | ||||||
| import { useAttendance } from "../../hooks/useAttendance"; | import { useAttendance } from "../../hooks/useAttendance"; // This hook is already providing data | ||||||
| import { useSelector } from "react-redux"; | import { useSelector } from "react-redux"; | ||||||
| import { useQueryClient } from "@tanstack/react-query"; | import { useQueryClient } from "@tanstack/react-query"; | ||||||
| import eventBus from "../../services/eventBus"; | import eventBus from "../../services/eventBus"; | ||||||
| 
 | 
 | ||||||
| const Attendance = ({ getRole, handleModalData }) => { | const Attendance = ({ getRole, handleModalData, attendance: filteredAndSearchedAttendanceFromParent, showOnlyCheckout, setshowOnlyCheckout }) => { | ||||||
|   const queryClient = useQueryClient(); |   const queryClient = useQueryClient(); | ||||||
|   const [loading, setLoading] = useState(false); |  | ||||||
|   const navigate = useNavigate(); |   const navigate = useNavigate(); | ||||||
|   const [todayDate, setTodayDate] = useState(new Date()); |   const [todayDate, setTodayDate] = useState(new Date()); | ||||||
|   const [ShowPending, setShowPending] = useState(false); | 
 | ||||||
|   const selectedProject = useSelector( |   const selectedProject = useSelector( | ||||||
|     (store) => store.localVariables.projectId |     (store) => store.localVariables.projectId | ||||||
|   ); |   ); | ||||||
|   const { |   const { | ||||||
|     attendance, |  | ||||||
|     loading: attLoading, |     loading: attLoading, | ||||||
|     recall: attrecall, |     recall: attrecall, | ||||||
|     isFetching |     isFetching | ||||||
|   } = useAttendance(selectedProject); |   } = useAttendance(selectedProject); // Keep this hook to manage recall and fetching status | ||||||
|   const filteredAttendance = ShowPending |  | ||||||
|     ? attendance?.filter( |  | ||||||
|         (att) => att?.checkInTime !== null && att?.checkOutTime === null |  | ||||||
|       ) |  | ||||||
|     : attendance; |  | ||||||
| 
 | 
 | ||||||
|   const attendanceList = Array.isArray(filteredAttendance) |   const attendanceList = Array.isArray(filteredAndSearchedAttendanceFromParent) | ||||||
|     ? filteredAttendance |     ? filteredAndSearchedAttendanceFromParent | ||||||
|     : []; |     : []; | ||||||
| 
 | 
 | ||||||
|   const sortByName = (a, b) => { |   const sortByName = (a, b) => { | ||||||
| @ -41,6 +34,7 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|     const nameB = (b.firstName + b.lastName).toLowerCase(); |     const nameB = (b.firstName + b.lastName).toLowerCase(); | ||||||
|     return nameA?.localeCompare(nameB); |     return nameA?.localeCompare(nameB); | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|   const group1 = attendanceList |   const group1 = attendanceList | ||||||
|     .filter((d) => d.activity === 1 || d.activity === 4) |     .filter((d) => d.activity === 1 || d.activity === 4) | ||||||
|     .sort(sortByName); |     .sort(sortByName); | ||||||
| @ -48,41 +42,39 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|     .filter((d) => d.activity === 0) |     .filter((d) => d.activity === 0) | ||||||
|     .sort(sortByName); |     .sort(sortByName); | ||||||
| 
 | 
 | ||||||
|   const filteredData = [...group1, ...group2]; |   const finalFilteredDataForPagination = [...group1, ...group2]; | ||||||
|  | 
 | ||||||
|   const { currentPage, totalPages, currentItems, paginate } = usePagination( |   const { currentPage, totalPages, currentItems, paginate } = usePagination( | ||||||
|     filteredData, |     finalFilteredDataForPagination, // Use the data that's already been searched and grouped | ||||||
|     ITEMS_PER_PAGE |     ITEMS_PER_PAGE | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   const handler = useCallback( |   const handler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
|       if (selectedProject == msg.projectId) { |       if (selectedProject === msg.projectId) { | ||||||
|         // const updatedAttendance = attendances.map((item) => |  | ||||||
|         //   item.employeeId === msg.response.employeeId |  | ||||||
|         //     ? { ...item, ...msg.response } |  | ||||||
|         //     : item |  | ||||||
|         // ); |  | ||||||
|         queryClient.setQueryData(["attendance", selectedProject], (oldData) => { |         queryClient.setQueryData(["attendance", selectedProject], (oldData) => { | ||||||
|           if (!oldData) { |           if (!oldData) { | ||||||
|             queryClient.invalidateQueries({queryKey:["attendance"]}) |             queryClient.invalidateQueries({ queryKey: ["attendance"] }); | ||||||
|           }; |             return; // Exit to avoid mapping on undefined oldData | ||||||
|  |           } | ||||||
|           return oldData.map((record) => |           return oldData.map((record) => | ||||||
|             record.employeeId === msg.response.employeeId ? { ...record, ...msg.response } : record |             record.employeeId === msg.response.employeeId ? { ...record, ...msg.response } : record | ||||||
|           ); |           ); | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     [selectedProject, attrecall] |     [selectedProject, queryClient] // Added queryClient to dependencies | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   const employeeHandler = useCallback( |   const employeeHandler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
|       if (attendances.some((item) => item.employeeId == msg.employeeId)) { |       if (attrecall) { // Check if attrecall function exists | ||||||
|         attrecall(); |         attrecall(); | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     [selectedProject, attendance] |     [attrecall] // Dependency should be attrecall, not `selectedProject` or `attendance` here | ||||||
|   ); |   ); | ||||||
|  | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     eventBus.on("attendance", handler); |     eventBus.on("attendance", handler); | ||||||
|     return () => eventBus.off("attendance", handler); |     return () => eventBus.off("attendance", handler); | ||||||
| @ -105,13 +97,14 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|               role="switch" |               role="switch" | ||||||
|               id="inactiveEmployeesCheckbox" |               id="inactiveEmployeesCheckbox" | ||||||
|               disabled={isFetching} |               disabled={isFetching} | ||||||
|               checked={ShowPending} |               checked={showOnlyCheckout} // Use prop for checked state | ||||||
|               onChange={(e) => setShowPending(e.target.checked)} |               onChange={(e) => setshowOnlyCheckout(e.target.checked)} // Use prop for onChange | ||||||
|             /> |             /> | ||||||
|             <label className="form-check-label ms-0">Show Pending</label> |             <label className="form-check-label ms-0">Show Pending</label> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         {Array.isArray(attendance) && attendance.length > 0 ? ( |         {/* Use `filteredAndSearchedAttendanceFromParent` for the initial check of data presence */} | ||||||
|  |         {Array.isArray(filteredAndSearchedAttendanceFromParent) && filteredAndSearchedAttendanceFromParent.length > 0 ? ( | ||||||
|           <> |           <> | ||||||
|             <table className="table "> |             <table className="table "> | ||||||
|               <thead> |               <thead> | ||||||
| @ -129,7 +122,7 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|                 </tr> |                 </tr> | ||||||
|               </thead> |               </thead> | ||||||
|               <tbody className="table-border-bottom-0 "> |               <tbody className="table-border-bottom-0 "> | ||||||
|                 {currentItems && |                 {currentItems && currentItems.length > 0 ? ( // Check currentItems length before mapping | ||||||
|                   currentItems |                   currentItems | ||||||
|                     .sort((a, b) => { |                     .sort((a, b) => { | ||||||
|                       const checkInA = a?.checkInTime |                       const checkInA = a?.checkInTime | ||||||
| @ -186,14 +179,18 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|                           /> |                           /> | ||||||
|                         </td> |                         </td> | ||||||
|                       </tr> |                       </tr> | ||||||
|                     ))} |                     )) | ||||||
|                 {!attendance && ( |                 ) : ( | ||||||
|                   <span className="text-secondary m-4">No employees assigned to the project!</span> |                   <tr> | ||||||
|  |                     <td colSpan="6" className="text-center text-muted py-4"> | ||||||
|  |                       No matching records found. | ||||||
|  |                     </td> | ||||||
|  |                   </tr> | ||||||
|                 )} |                 )} | ||||||
|               </tbody> |               </tbody> | ||||||
|             </table> |             </table> | ||||||
| 
 | 
 | ||||||
|             {!loading && filteredData.length > 20 && ( |             {!attLoading && finalFilteredDataForPagination.length > ITEMS_PER_PAGE && ( // Use the data before pagination for total count check | ||||||
|               <nav aria-label="Page "> |               <nav aria-label="Page "> | ||||||
|                 <ul className="pagination pagination-sm justify-content-end py-1"> |                 <ul className="pagination pagination-sm justify-content-end py-1"> | ||||||
|                   <li |                   <li | ||||||
| @ -243,14 +240,16 @@ const Attendance = ({ getRole, handleModalData }) => { | |||||||
|           <div>Loading...</div> |           <div>Loading...</div> | ||||||
|         ) : ( |         ) : ( | ||||||
|           <div className="text-muted"> |           <div className="text-muted"> | ||||||
|             {Array.isArray(attendance) |             {/* Check the actual prop passed for initial data presence */} | ||||||
|               ? "No employees assigned to the project" |             {Array.isArray(filteredAndSearchedAttendanceFromParent) && filteredAndSearchedAttendanceFromParent.length === 0 | ||||||
|               : "Attendance data unavailable"} |               ? "" | ||||||
|  |               : "Attendance data unavailable."} | ||||||
|           </div> |           </div> | ||||||
|         )} |         )} | ||||||
| 
 | 
 | ||||||
|         {currentItems?.length == 0 && attendance.length > 0 && ( |         {/* This condition should check `currentItems` or `finalFilteredDataForPagination` */} | ||||||
|           <div className="my-4"><span className="text-secondary">No Pending Record Available !</span></div> |         {currentItems?.length === 0 && finalFilteredDataForPagination.length > 0 && showOnlyCheckout && ( | ||||||
|  |           <div className="my-4"><span className="text-secondary">No Pending Record Available for your search!</span></div> | ||||||
|         )} |         )} | ||||||
|       </div> |       </div> | ||||||
|     </> |     </> | ||||||
|  | |||||||
| @ -4,24 +4,31 @@ 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, setAttendanceData } from "../../slices/apiSlice/attedanceLogsSlice"; | ||||||
| import DateRangePicker from "../common/DateRangePicker"; | import DateRangePicker from "../common/DateRangePicker"; | ||||||
| import { clearCacheKey, getCachedData } from "../../slices/apiDataManager"; |  | ||||||
| import eventBus from "../../services/eventBus"; | import eventBus from "../../services/eventBus"; | ||||||
| import AttendanceRepository from "../../repositories/AttendanceRepository"; |  | ||||||
| import { useAttendancesLogs } from "../../hooks/useAttendance"; |  | ||||||
| import { queryClient } from "../../layouts/AuthLayout"; |  | ||||||
| 
 | 
 | ||||||
| const usePagination = (data, itemsPerPage) => { | const usePagination = (data, itemsPerPage) => { | ||||||
|   const [currentPage, setCurrentPage] = useState(1); |   const [currentPage, setCurrentPage] = useState(1); | ||||||
|   const maxPage = Math.ceil(data.length / itemsPerPage); |   const totalItems = Array.isArray(data) ? data.length : 0; | ||||||
|  |   const maxPage = Math.ceil(totalItems / itemsPerPage); | ||||||
|  | 
 | ||||||
|   const currentItems = useMemo(() => { |   const currentItems = useMemo(() => { | ||||||
|  |     if (!Array.isArray(data) || data.length === 0) { | ||||||
|  |       return []; | ||||||
|  |     } | ||||||
|     const startIndex = (currentPage - 1) * itemsPerPage; |     const startIndex = (currentPage - 1) * itemsPerPage; | ||||||
|     const endIndex = startIndex + itemsPerPage; |     const endIndex = startIndex + itemsPerPage; | ||||||
|     return data.slice(startIndex, endIndex); |     return data.slice(startIndex, endIndex); | ||||||
|   }, [data, currentPage, itemsPerPage]); |   }, [data, currentPage, itemsPerPage]); | ||||||
| 
 | 
 | ||||||
|   const paginate = useCallback((pageNumber) => setCurrentPage(pageNumber), []); |   const paginate = useCallback((pageNumber) => { | ||||||
|  |     if (pageNumber > 0 && pageNumber <= maxPage) { | ||||||
|  |       setCurrentPage(pageNumber); | ||||||
|  |     } | ||||||
|  |   }, [maxPage]); | ||||||
|  | 
 | ||||||
|  |   // Ensure resetPage is returned by the hook | ||||||
|   const resetPage = useCallback(() => setCurrentPage(1), []); |   const resetPage = useCallback(() => setCurrentPage(1), []); | ||||||
| 
 | 
 | ||||||
|   return { |   return { | ||||||
| @ -35,60 +42,91 @@ const usePagination = (data, itemsPerPage) => { | |||||||
| 
 | 
 | ||||||
| const AttendanceLog = ({ | const AttendanceLog = ({ | ||||||
|   handleModalData, |   handleModalData, | ||||||
|  |   projectId, | ||||||
|  |   setshowOnlyCheckout, | ||||||
|  |   showOnlyCheckout, | ||||||
|  |   searchQuery, // Prop for search query | ||||||
| }) => { | }) => { | ||||||
|   const selectedProject = useSelector( |  | ||||||
|     (store) => store.localVariables.projectId |  | ||||||
|   ); |  | ||||||
|   const [dateRange, setDateRange] = useState({ startDate: "", endDate: "" }); |   const [dateRange, setDateRange] = useState({ startDate: "", endDate: "" }); | ||||||
|   const dispatch = useDispatch(); |   const dispatch = useDispatch(); | ||||||
|   const [loading, setLoading] = useState(false); |   const { data, loading, error } = useSelector((store) => store.attendanceLogs); | ||||||
|   const [showPending,setShowPending] = useState(false) |  | ||||||
| 
 |  | ||||||
|   const [isRefreshing, setIsRefreshing] = useState(false); |   const [isRefreshing, setIsRefreshing] = useState(false); | ||||||
|   const [processedData, setProcessedData] = useState([]); |  | ||||||
| 
 | 
 | ||||||
|   const today = new Date(); |   const today = useMemo(() => { | ||||||
|   today.setHours(0, 0, 0, 0); |     const d = new Date(); | ||||||
|  |     d.setHours(0, 0, 0, 0); | ||||||
|  |     return d; | ||||||
|  |   }, []); | ||||||
| 
 | 
 | ||||||
|   const yesterday = new Date(); |   const yesterday = useMemo(() => { | ||||||
|   yesterday.setDate(yesterday.getDate() - 1); |     const d = new Date(); | ||||||
|  |     d.setDate(d.getDate() - 1); | ||||||
|  |     return d; | ||||||
|  |   }, []); | ||||||
| 
 | 
 | ||||||
|   const isSameDay = (dateStr) => { |   const isSameDay = useCallback((dateStr) => { | ||||||
|     if (!dateStr) return false; |     if (!dateStr) return false; | ||||||
|     const d = new Date(dateStr); |     const d = new Date(dateStr); | ||||||
|     d.setHours(0, 0, 0, 0); |     d.setHours(0, 0, 0, 0); | ||||||
|     return d.getTime() === today.getTime(); |     return d.getTime() === today.getTime(); | ||||||
|   }; |   }, [today]); | ||||||
| 
 | 
 | ||||||
|   const isBeforeToday = (dateStr) => { |   const isBeforeToday = useCallback((dateStr) => { | ||||||
|     if (!dateStr) return false; |     if (!dateStr) return false; | ||||||
|     const d = new Date(dateStr); |     const d = new Date(dateStr); | ||||||
|     d.setHours(0, 0, 0, 0); |     d.setHours(0, 0, 0, 0); | ||||||
|     return d.getTime() < today.getTime(); |     return d.getTime() < today.getTime(); | ||||||
|   }; |   }, [today]); | ||||||
| 
 | 
 | ||||||
|   const sortByName = (a, b) => { |   const sortByName = useCallback((a, b) => { | ||||||
|     const nameA = a.firstName.toLowerCase() + a.lastName.toLowerCase(); |     const nameA = `${a.firstName || ""} ${a.lastName || ""}`.toLowerCase(); | ||||||
|     const nameB = b.firstName.toLowerCase() + b.lastName.toLowerCase(); |     const nameB = `${b.firstName || ""} ${b.lastName || ""}`.toLowerCase(); | ||||||
|     return nameA?.localeCompare(nameB); |     return nameA.localeCompare(nameB); | ||||||
|   }; |   }, []); | ||||||
| 
 | 
 | ||||||
|   const { |   useEffect(() => { | ||||||
|     data = [], |     const { startDate, endDate } = dateRange; | ||||||
|     isLoading, |     dispatch( | ||||||
|     error, |       fetchAttendanceData({ | ||||||
|     refetch, |         projectId, | ||||||
|     isFetching, |         fromDate: startDate, | ||||||
|   } = useAttendancesLogs( |         toDate: endDate, | ||||||
|     selectedProject, |       }) | ||||||
|     dateRange.startDate, |  | ||||||
|     dateRange.endDate |  | ||||||
|     ); |     ); | ||||||
|   const filtering = (data) => { |     setIsRefreshing(false); | ||||||
|     const filteredData = showPending |   }, [dateRange, projectId, dispatch, isRefreshing]); | ||||||
|  | 
 | ||||||
|  |   const processedData = useMemo(() => { | ||||||
|  |     let filteredData = showOnlyCheckout | ||||||
|       ? data.filter((item) => item.checkOutTime === null) |       ? data.filter((item) => item.checkOutTime === null) | ||||||
|       : data; |       : data; | ||||||
| 
 | 
 | ||||||
|  |     // Apply search query filter | ||||||
|  |     if (searchQuery) { | ||||||
|  |       const lowerCaseSearchQuery = searchQuery.toLowerCase().trim(); // Trim whitespace | ||||||
|  | 
 | ||||||
|  |       filteredData = filteredData.filter((att) => { | ||||||
|  |         // Option 1: Combine firstName, middleName, lastName | ||||||
|  |         const fullName = [att.firstName, att.middleName, att.lastName] | ||||||
|  |           .filter(Boolean) // This removes null, undefined, or empty string parts | ||||||
|  |           .join(" ") | ||||||
|  |           .toLowerCase(); | ||||||
|  | 
 | ||||||
|  |         // Option 2: Check `employeeName` if it exists and is reliable | ||||||
|  |         const employeeName = att.employeeName?.toLowerCase() || ""; | ||||||
|  | 
 | ||||||
|  |         // Option 3: Check `employeeId` | ||||||
|  |         const employeeId = att.employeeId?.toLowerCase() || ""; | ||||||
|  | 
 | ||||||
|  |         // Check if the search query is included in any of the relevant fields | ||||||
|  |         return ( | ||||||
|  |           fullName.includes(lowerCaseSearchQuery) || | ||||||
|  |           employeeName.includes(lowerCaseSearchQuery) || | ||||||
|  |           employeeId.includes(lowerCaseSearchQuery) | ||||||
|  |         ); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     const group1 = filteredData |     const group1 = filteredData | ||||||
|       .filter((d) => d.activity === 1 && isSameDay(d.checkInTime)) |       .filter((d) => d.activity === 1 && isSameDay(d.checkInTime)) | ||||||
|       .sort(sortByName); |       .sort(sortByName); | ||||||
| @ -127,53 +165,46 @@ const AttendanceLog = ({ | |||||||
|       return acc; |       return acc; | ||||||
|     }, {}); |     }, {}); | ||||||
| 
 | 
 | ||||||
|  |     // Sort dates in descending order | ||||||
|     const sortedDates = Object.keys(groupedByDate).sort( |     const sortedDates = Object.keys(groupedByDate).sort( | ||||||
|       (a, b) => new Date(b) - new Date(a) |       (a, b) => new Date(b) - new Date(a) | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     const finalData = sortedDates.flatMap((date) => groupedByDate[date]); |     // Create the final sorted array | ||||||
|     setProcessedData(finalData); |     return sortedDates.flatMap((date) => groupedByDate[date]); | ||||||
|   }; |   }, [data, showOnlyCheckout, searchQuery, isSameDay, isBeforeToday, sortByName]); | ||||||
| 
 |  | ||||||
|   useEffect(() => { |  | ||||||
|     filtering(data); |  | ||||||
|   }, [data, showPending]); |  | ||||||
| 
 | 
 | ||||||
|   const { |   const { | ||||||
|     currentPage, |     currentPage, | ||||||
|     totalPages, |     totalPages, | ||||||
|     currentItems: paginatedAttendances, |     currentItems: paginatedAttendances, | ||||||
|     paginate, |     paginate, | ||||||
|     resetPage, |     resetPage, // Destructure resetPage here | ||||||
|   } = usePagination(processedData, 20); |   } = usePagination(processedData, 20); | ||||||
| 
 | 
 | ||||||
|  |   // Effect to reset pagination when search query changes | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     resetPage(); |     resetPage(); | ||||||
|   }, [processedData, resetPage]); |   }, [searchQuery, resetPage]); // Add resetPage to dependencies | ||||||
| 
 | 
 | ||||||
|   const handler = useCallback( |   const handler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
|       const { startDate, endDate } = dateRange; |       const { startDate, endDate } = dateRange; | ||||||
|       const checkIn = msg.response.checkInTime.substring(0, 10); |       const checkIn = msg.response.checkInTime.substring(0, 10); | ||||||
|       if ( |       if ( | ||||||
|         selectedProject === msg.projectId && |         projectId === msg.projectId && | ||||||
|         startDate <= checkIn && |         startDate <= checkIn && | ||||||
|         checkIn <= endDate |         checkIn <= endDate | ||||||
|       ) { |       ) { | ||||||
|         queryClient.setQueriesData(["attendanceLogs"],(oldData)=>{ |         const updatedAttendance = data.map((item) => | ||||||
|           if(!oldData) { |           item.id === msg.response.id | ||||||
|             queryClient.invalidateQueries({queryKey:["attendanceLogs"]}) |             ? { ...item, ...msg.response } | ||||||
|           } |             : item | ||||||
|           return oldData.map((record) => |  | ||||||
|           record.id === msg.response.id ? { ...record, ...msg.response } : record |  | ||||||
|         ); |         ); | ||||||
|         }) |         dispatch(setAttendanceData(updatedAttendance)); // Update Redux store | ||||||
| 
 |  | ||||||
|         filtering(updatedAttendance); |  | ||||||
|         resetPage(); |  | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     [selectedProject, dateRange, data, filtering, resetPage] |     [projectId, dateRange, data, dispatch] | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
| @ -184,19 +215,15 @@ const AttendanceLog = ({ | |||||||
|   const employeeHandler = useCallback( |   const employeeHandler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
|       const { startDate, endDate } = dateRange; |       const { startDate, endDate } = dateRange; | ||||||
|       if (data.some((item) => item.employeeId == msg.employeeId)) { |       dispatch( | ||||||
|         // dispatch( |         fetchAttendanceData({ | ||||||
|         //   fetchAttendanceData({ |           projectId, | ||||||
|         //     , |           fromDate: startDate, | ||||||
|         //     fromDate: startDate, |           toDate: endDate, | ||||||
|         //     toDate: endDate, |         }) | ||||||
|         //   }) |       ); | ||||||
|         // ); |  | ||||||
| 
 |  | ||||||
|         refetch() |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
|     [selectedProject, dateRange, data] |     [projectId, dateRange, dispatch] | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
| @ -220,28 +247,27 @@ const AttendanceLog = ({ | |||||||
|               type="checkbox" |               type="checkbox" | ||||||
|               className="form-check-input" |               className="form-check-input" | ||||||
|               role="switch" |               role="switch" | ||||||
|               disabled={isFetching} |  | ||||||
|               id="inactiveEmployeesCheckbox" |               id="inactiveEmployeesCheckbox" | ||||||
|               checked={showPending} |               checked={showOnlyCheckout} | ||||||
|               onChange={(e) => setShowPending(e.target.checked)} |               onChange={(e) => setshowOnlyCheckout(e.target.checked)} | ||||||
|             /> |             /> | ||||||
|             <label className="form-check-label ms-0">Show Pending</label> |             <label className="form-check-label ms-0">Show Pending</label> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div className="col-md-2 m-0 text-end"> |         <div className="col-md-2 m-0 text-end"> | ||||||
|           <i |           <i | ||||||
|             className={`bx bx-refresh cursor-pointer fs-4 ${ |             className={`bx bx-refresh cursor-pointer fs-4 ${loading || isRefreshing ? "spin" : "" | ||||||
|               isFetching ? "spin" : "" |  | ||||||
|               }`} |               }`} | ||||||
|             title="Refresh" |             title="Refresh" | ||||||
|             onClick={() => refetch()} |             onClick={() => setIsRefreshing(true)} | ||||||
|           /> |           /> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|       <div className="table-responsive text-nowrap"> |       <div | ||||||
|         {isLoading ? ( |         className="table-responsive text-nowrap" | ||||||
|           <div><p className="text-secondary">Loading...</p></div> |         style={{ minHeight: "200px", display: 'flex' }} | ||||||
|         ) : data?.length > 0 ? ( |       > | ||||||
|  |         {processedData && processedData.length > 0 ? ( | ||||||
|           <table className="table mb-0"> |           <table className="table mb-0"> | ||||||
|             <thead> |             <thead> | ||||||
|               <tr> |               <tr> | ||||||
| @ -260,7 +286,14 @@ const AttendanceLog = ({ | |||||||
|               </tr> |               </tr> | ||||||
|             </thead> |             </thead> | ||||||
|             <tbody> |             <tbody> | ||||||
|               {paginatedAttendances.reduce((acc, attendance, index, arr) => { |               {(loading || isRefreshing) && ( | ||||||
|  |                 <tr> | ||||||
|  |                   <td colSpan={6}>Loading...</td> | ||||||
|  |                 </tr> | ||||||
|  |               )} | ||||||
|  |               {!loading && | ||||||
|  |                 !isRefreshing && | ||||||
|  |                 paginatedAttendances.reduce((acc, attendance, index, arr) => { | ||||||
|                   const currentDate = moment( |                   const currentDate = moment( | ||||||
|                     attendance.checkInTime || attendance.checkOutTime |                     attendance.checkInTime || attendance.checkOutTime | ||||||
|                   ).format("YYYY-MM-DD"); |                   ).format("YYYY-MM-DD"); | ||||||
| @ -287,7 +320,7 @@ const AttendanceLog = ({ | |||||||
|                     ); |                     ); | ||||||
|                   } |                   } | ||||||
|                   acc.push( |                   acc.push( | ||||||
|                   <tr key={index}> |                     <tr key={attendance.id || 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 |                           <Avatar | ||||||
| @ -329,13 +362,20 @@ const AttendanceLog = ({ | |||||||
|             </tbody> |             </tbody> | ||||||
|           </table> |           </table> | ||||||
|         ) : ( |         ) : ( | ||||||
|            <div className="my-4"><span className="text-secondary">No Record Available !</span></div> |           !loading && | ||||||
|  |           !isRefreshing && ( | ||||||
|  |             <div | ||||||
|  |               className="d-flex justify-content-center align-items-center text-muted" | ||||||
|  |               style={{ | ||||||
|  |                 width: "100%", | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               No employee logs. | ||||||
|  |             </div> | ||||||
|  |           ) | ||||||
|         )} |         )} | ||||||
|       </div> |       </div> | ||||||
|         {paginatedAttendances?.length == 0 && data?.length > 0 && ( |       {!loading && !isRefreshing && processedData.length > 20 && ( | ||||||
|           <div className="my-4"><span className="text-secondary">No Pending Record Available !</span></div> |  | ||||||
|         )} |  | ||||||
|       {processedData.length > 10 && ( |  | ||||||
|         <nav aria-label="Page "> |         <nav aria-label="Page "> | ||||||
|           <ul className="pagination pagination-sm justify-content-end py-1"> |           <ul className="pagination pagination-sm justify-content-end py-1"> | ||||||
|             <li className={`page-item ${currentPage === 1 ? "disabled" : ""}`}> |             <li className={`page-item ${currentPage === 1 ? "disabled" : ""}`}> | ||||||
| @ -350,8 +390,7 @@ const AttendanceLog = ({ | |||||||
|               (pageNumber) => ( |               (pageNumber) => ( | ||||||
|                 <li |                 <li | ||||||
|                   key={pageNumber} |                   key={pageNumber} | ||||||
|                   className={`page-item ${ |                   className={`page-item ${currentPage === pageNumber ? "active" : "" | ||||||
|                     currentPage === pageNumber ? "active" : "" |  | ||||||
|                     }`} |                     }`} | ||||||
|                 > |                 > | ||||||
|                   <button |                   <button | ||||||
| @ -364,8 +403,7 @@ const AttendanceLog = ({ | |||||||
|               ) |               ) | ||||||
|             )} |             )} | ||||||
|             <li |             <li | ||||||
|               className={`page-item ${ |               className={`page-item ${currentPage === totalPages ? "disabled" : "" | ||||||
|                 currentPage === totalPages ? "disabled" : "" |  | ||||||
|                 }`} |                 }`} | ||||||
|             > |             > | ||||||
|               <button |               <button | ||||||
|  | |||||||
| @ -11,10 +11,43 @@ import { checkIfCurrentDate } from "../../utils/dateUtils"; | |||||||
| import { useMarkAttendance } from "../../hooks/useAttendance"; | import { useMarkAttendance } from "../../hooks/useAttendance"; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| const schema = z.object({ | // const schema = z.object({ | ||||||
|  | //   markTime: z.string().nonempty({ message: "Time is required" }), | ||||||
|  | //   description: z.string().max(200, "description should less than 200 chracters").optional() | ||||||
|  | // }); | ||||||
|  | 
 | ||||||
|  | const createSchema = (modeldata) => { | ||||||
|  |   return z | ||||||
|  |     .object({ | ||||||
|       markTime: z.string().nonempty({ message: "Time is required" }), |       markTime: z.string().nonempty({ message: "Time is required" }), | ||||||
|   description: z.string().max(200, "description should less than 200 chracters").optional() |       description: z | ||||||
|  |         .string() | ||||||
|  |         .max(200, "Description should be less than 200 characters") | ||||||
|  |         .optional(), | ||||||
|  |     }) | ||||||
|  |     .refine((data) => { | ||||||
|  |       if (modeldata?.checkInTime && !modeldata?.checkOutTime) { | ||||||
|  |         const checkIn = new Date(modeldata.checkInTime); | ||||||
|  |         const [time, modifier] = data.markTime.split(" "); | ||||||
|  |         const [hourStr, minuteStr] = time.split(":"); | ||||||
|  |         let hour = parseInt(hourStr, 10); | ||||||
|  |         const minute = parseInt(minuteStr, 10); | ||||||
|  | 
 | ||||||
|  |         if (modifier === "PM" && hour !== 12) hour += 12; | ||||||
|  |         if (modifier === "AM" && hour === 12) hour = 0; | ||||||
|  | 
 | ||||||
|  |         const checkOut = new Date(checkIn); | ||||||
|  |         checkOut.setHours(hour, minute, 0, 0); | ||||||
|  | 
 | ||||||
|  |         return checkOut > checkIn; | ||||||
|  |       } | ||||||
|  |       return true; | ||||||
|  |     }, { | ||||||
|  |       message: "Checkout time must be later than check-in time", | ||||||
|  |       path: ["markTime"], | ||||||
|     }); |     }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| const CheckCheckOutmodel = ({ modeldata, closeModal, handleSubmitForm, }) => { | const CheckCheckOutmodel = ({ modeldata, closeModal, handleSubmitForm, }) => { | ||||||
| 
 | 
 | ||||||
| @ -33,6 +66,17 @@ const CheckCheckOutmodel = ({ modeldata, closeModal, handleSubmitForm, }) => { | |||||||
|     return `${day}-${month}-${year}`; |     return `${day}-${month}-${year}`; | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   // const { | ||||||
|  |   //   register, | ||||||
|  |   //   handleSubmit, | ||||||
|  |   //   formState: { errors }, | ||||||
|  |   //   reset, | ||||||
|  |   //   setValue, | ||||||
|  |   // } = useForm({ | ||||||
|  |   //   resolver: zodResolver(schema), | ||||||
|  |   //   mode: "onChange" | ||||||
|  |   // }); | ||||||
|  | 
 | ||||||
|   const { |   const { | ||||||
|   register, |   register, | ||||||
|   handleSubmit, |   handleSubmit, | ||||||
| @ -40,31 +84,20 @@ const CheckCheckOutmodel = ({ modeldata, closeModal, handleSubmitForm, }) => { | |||||||
|   reset, |   reset, | ||||||
|   setValue, |   setValue, | ||||||
| } = useForm({ | } = useForm({ | ||||||
|     resolver: zodResolver(schema), |   resolver: zodResolver(createSchema(modeldata)), | ||||||
|     mode: "onChange" |   mode: "onChange", | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|   const onSubmit = (data) => { |   const onSubmit = (data) => { | ||||||
|     let record = { ...data, date: new Date().toLocaleDateString(), latitude: coords.latitude, longitude: coords.longitude, employeeId: modeldata.employeeId, action: modeldata.action, id: modeldata?.id || null } |     let record = { ...data, date: new Date().toLocaleDateString(), latitude: coords.latitude, longitude: coords.longitude, employeeId: modeldata.employeeId, action: modeldata.action, id: modeldata?.id || null } | ||||||
|     // if (modeldata.forWhichTab === 1 || modeldata.forWhichTab === 2) { |     if (modeldata.forWhichTab === 1) { | ||||||
|       // handleSubmitForm(record) |       handleSubmitForm(record) | ||||||
|           const payload = { |     } else { | ||||||
|           Id: modeldata?.id || null, | 
 | ||||||
|           comment: data.description, |       dispatch(markAttendance(record)) | ||||||
|           employeeID: modeldata.employeeId, |         .unwrap() | ||||||
|           projectId: projectId, |         .then((data) => { | ||||||
|           date: new Date().toISOString(), |  | ||||||
|           markTime: data.markTime, |  | ||||||
|           latitude: coords.latitude.toString(), |  | ||||||
|           longitude: coords.longitude.toString(), |  | ||||||
|           action: modeldata.action, |  | ||||||
|           image: null, |  | ||||||
|         }; |  | ||||||
|       MarkAttendance({payload,forWhichTab:modeldata.forWhichTab}) |  | ||||||
|     // } else { |  | ||||||
|     //   dispatch(markAttendance(record)) |  | ||||||
|     //     .unwrap() |  | ||||||
|     //     .then((data) => { |  | ||||||
| 
 | 
 | ||||||
|     //       showToast("Attendance Marked Successfully", "success"); |     //       showToast("Attendance Marked Successfully", "success"); | ||||||
|     //     }) |     //     }) | ||||||
| @ -72,8 +105,8 @@ const CheckCheckOutmodel = ({ modeldata, closeModal, handleSubmitForm, }) => { | |||||||
| 
 | 
 | ||||||
|     //       showToast(error, "error"); |     //       showToast(error, "error"); | ||||||
| 
 | 
 | ||||||
|     //     }); |         }); | ||||||
|     // } |     } | ||||||
| 
 | 
 | ||||||
|     closeModal() |     closeModal() | ||||||
|   }; |   }; | ||||||
|  | |||||||
| @ -7,63 +7,37 @@ import { useRegularizationRequests } from "../../hooks/useAttendance"; | |||||||
| import moment from "moment"; | import moment from "moment"; | ||||||
| import usePagination from "../../hooks/usePagination"; | import usePagination from "../../hooks/usePagination"; | ||||||
| import eventBus from "../../services/eventBus"; | import eventBus from "../../services/eventBus"; | ||||||
| import { cacheData, clearCacheKey } from "../../slices/apiDataManager"; | import { cacheData } from "../../slices/apiDataManager"; | ||||||
| import { useQueryClient } from "@tanstack/react-query"; |  | ||||||
| 
 | 
 | ||||||
| const Regularization = ({ handleRequest }) => { | const Regularization = ({ handleRequest, searchQuery }) => { | ||||||
|   const queryClient = useQueryClient(); |   const selectedProject = useSelector((store) => store.localVariables.projectId); | ||||||
|   var selectedProject = useSelector((store) => store.localVariables.projectId); |   const [regularizesList, setRegularizedList] = useState([]); | ||||||
|   const [regularizesList, setregularizedList] = useState([]); |   const { regularizes, loading, refetch } = useRegularizationRequests(selectedProject); | ||||||
|   const { regularizes, loading, error, refetch } = |  | ||||||
|     useRegularizationRequests(selectedProject); |  | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     setregularizedList(regularizes); |     setRegularizedList(regularizes); | ||||||
|   }, [regularizes]); |   }, [regularizes]); | ||||||
| 
 | 
 | ||||||
|   const sortByName = (a, b) => { |   const sortByName = (a, b) => { | ||||||
|     const nameA = a.firstName.toLowerCase() + a.lastName.toLowerCase(); |     const nameA = (a.firstName + a.lastName).toLowerCase(); | ||||||
|     const nameB = b.firstName.toLowerCase() + b.lastName.toLowerCase(); |     const nameB = (b.firstName + b.lastName).toLowerCase(); | ||||||
|     return nameA?.localeCompare(nameB); |     return nameA.localeCompare(nameB); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const handler = useCallback( |   const handler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
|       if (selectedProject == msg.projectId) { |       if (selectedProject == msg.projectId) { | ||||||
|         // const updatedAttendance = regularizes?.filter( |         const updatedAttendance = regularizes?.filter(item => item.id !== msg.response.id); | ||||||
|         //   (item) => item.id !== msg.response.id |         cacheData("regularizedList", { | ||||||
|         // ); |           data: updatedAttendance, | ||||||
|         // cacheData("regularizedList", { |           projectId: selectedProject, | ||||||
|         //   data: updatedAttendance, |         }); | ||||||
|         //   projectId: selectedProject, |         refetch(); | ||||||
|         // }); |  | ||||||
|         // refetch(); |  | ||||||
| 
 |  | ||||||
|         queryClient.setQueryData( |  | ||||||
|           ["regularizedList", selectedProject], |  | ||||||
|           (oldData) => { |  | ||||||
|             if (!oldData) { |  | ||||||
|               queryClient.invalidateQueries({ queryKey: ["regularizedList"] }); |  | ||||||
|             } |  | ||||||
|             return oldData.filter((record) => record.id !== msg.response.id); |  | ||||||
|           } |  | ||||||
|         ), |  | ||||||
|           queryClient.invalidateQueries({ queryKey: ["attendanceLogs"] }); |  | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     [selectedProject, regularizes] |     [selectedProject, regularizes] | ||||||
|   ); |   ); | ||||||
|    |    | ||||||
|   const filteredData = [...regularizesList]?.sort(sortByName); |  | ||||||
| 
 |  | ||||||
|   const { currentPage, totalPages, currentItems, paginate } = usePagination( |  | ||||||
|     filteredData, |  | ||||||
|     20 |  | ||||||
|   ); |  | ||||||
|   useEffect(() => { |  | ||||||
|     eventBus.on("regularization", handler); |  | ||||||
|     return () => eventBus.off("regularization", handler); |  | ||||||
|   }, [handler]); |  | ||||||
| 
 | 
 | ||||||
|   const employeeHandler = useCallback( |   const employeeHandler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
| @ -74,18 +48,36 @@ const Regularization = ({ handleRequest }) => { | |||||||
|     [regularizes] |     [regularizes] | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     eventBus.on("regularization", handler); | ||||||
|  |     return () => eventBus.off("regularization", handler); | ||||||
|  |   }, [handler]); | ||||||
|  | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     eventBus.on("employee", employeeHandler); |     eventBus.on("employee", employeeHandler); | ||||||
|     return () => eventBus.off("employee", employeeHandler); |     return () => eventBus.off("employee", employeeHandler); | ||||||
|   }, [employeeHandler]); |   }, [employeeHandler]); | ||||||
| 
 | 
 | ||||||
|  |   // ✅ Search filter logic added here | ||||||
|  |   const filteredData = [...regularizesList] | ||||||
|  |     ?.filter((item) => { | ||||||
|  |       if (!searchQuery) return true; | ||||||
|  |       const lowerSearch = searchQuery.toLowerCase(); | ||||||
|  |       const fullName = `${item.firstName || ""} ${item.lastName || ""}`.toLowerCase(); | ||||||
|  | 
 | ||||||
|  |       return ( | ||||||
|  |         item.firstName?.toLowerCase().includes(lowerSearch) || | ||||||
|  |         item.lastName?.toLowerCase().includes(lowerSearch) || | ||||||
|  |         fullName.includes(lowerSearch) || | ||||||
|  |         item.employeeId?.toLowerCase().includes(lowerSearch) | ||||||
|  |       ); | ||||||
|  |     }) | ||||||
|  |     .sort(sortByName); | ||||||
|  | 
 | ||||||
|  |   const { currentPage, totalPages, currentItems, paginate } = usePagination(filteredData, 20); | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className="table-responsive text-nowrap pb-4"> |     <div className="table-responsive text-nowrap pb-4"> | ||||||
|       {loading ? ( |  | ||||||
|         <div className="my-2"> |  | ||||||
|           <p className="text-secondary">Loading...</p> |  | ||||||
|         </div> |  | ||||||
|       ) : currentItems?.length > 0 ? ( |  | ||||||
|       <table className="table mb-0"> |       <table className="table mb-0"> | ||||||
|         <thead> |         <thead> | ||||||
|           <tr> |           <tr> | ||||||
| @ -101,14 +93,12 @@ const Regularization = ({ handleRequest }) => { | |||||||
|           </tr> |           </tr> | ||||||
|         </thead> |         </thead> | ||||||
|         <tbody> |         <tbody> | ||||||
|             {currentItems?.map((att, index) => ( |           {!loading && currentItems?.length > 0 ? ( | ||||||
|  |             currentItems.map((att, 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 |                     <Avatar firstName={att.firstName} lastName={att.lastName} /> | ||||||
|                       firstName={att.firstName} |  | ||||||
|                       lastName={att.lastName} |  | ||||||
|                     ></Avatar> |  | ||||||
|                     <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-normal"> |                         <span className="fw-normal"> | ||||||
| @ -129,18 +119,27 @@ const Regularization = ({ handleRequest }) => { | |||||||
|                     handleRequest={handleRequest} |                     handleRequest={handleRequest} | ||||||
|                     refresh={refetch} |                     refresh={refetch} | ||||||
|                   /> |                   /> | ||||||
|                   {/* </div> */} |  | ||||||
|                 </td> |                 </td> | ||||||
|               </tr> |               </tr> | ||||||
|             ))} |             )) | ||||||
|  |           ) : ( | ||||||
|  |             <tr> | ||||||
|  |               <td | ||||||
|  |                 colSpan={6} | ||||||
|  |                 className="text-center" | ||||||
|  |                 style={{ | ||||||
|  |                   height: "200px", | ||||||
|  |                   verticalAlign: "middle", | ||||||
|  |                   borderBottom: "none", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 {loading ? "Loading..." : "No Record Found"} | ||||||
|  |               </td> | ||||||
|  |             </tr> | ||||||
|  |           )} | ||||||
|         </tbody> |         </tbody> | ||||||
|       </table> |       </table> | ||||||
|       ) : ( | 
 | ||||||
|         <div className="my-4"> |  | ||||||
|           {" "} |  | ||||||
|           <span className="text-secondary">No Requests Found !</span> |  | ||||||
|         </div> |  | ||||||
|       )} |  | ||||||
|       {!loading && totalPages > 1 && ( |       {!loading && totalPages > 1 && ( | ||||||
|         <nav aria-label="Page "> |         <nav aria-label="Page "> | ||||||
|           <ul className="pagination pagination-sm justify-content-end py-1 mt-3"> |           <ul className="pagination pagination-sm justify-content-end py-1 mt-3"> | ||||||
| @ -155,22 +154,15 @@ const Regularization = ({ handleRequest }) => { | |||||||
|             {[...Array(totalPages)].map((_, index) => ( |             {[...Array(totalPages)].map((_, index) => ( | ||||||
|               <li |               <li | ||||||
|                 key={index} |                 key={index} | ||||||
|                 className={`page-item ${ |                 className={`page-item ${currentPage === index + 1 ? "active" : ""}`} | ||||||
|                   currentPage === index + 1 ? "active" : "" |  | ||||||
|                 }`} |  | ||||||
|               > |  | ||||||
|                 <button |  | ||||||
|                   className="page-link " |  | ||||||
|                   onClick={() => paginate(index + 1)} |  | ||||||
|               > |               > | ||||||
|  |                 <button className="page-link" onClick={() => paginate(index + 1)}> | ||||||
|                   {index + 1} |                   {index + 1} | ||||||
|                 </button> |                 </button> | ||||||
|               </li> |               </li> | ||||||
|             ))} |             ))} | ||||||
|             <li |             <li | ||||||
|               className={`page-item ${ |               className={`page-item ${currentPage === totalPages ? "disabled" : ""}`} | ||||||
|                 currentPage === totalPages ? "disabled" : "" |  | ||||||
|               }`} |  | ||||||
|             > |             > | ||||||
|               <button |               <button | ||||||
|                 className="page-link" |                 className="page-link" | ||||||
|  | |||||||
| @ -5,7 +5,6 @@ import { useDashboardTasksCardData } from "../../hooks/useDashboard_Data"; | |||||||
| const TasksCard = () => { | const TasksCard = () => { | ||||||
|   const projectId = useSelector((store) => store.localVariables?.projectId); |   const projectId = useSelector((store) => store.localVariables?.projectId); | ||||||
|   const { tasksCardData, loading, error } = useDashboardTasksCardData(projectId); |   const { tasksCardData, loading, error } = useDashboardTasksCardData(projectId); | ||||||
|   console.log(tasksCardData); |  | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className="card p-3 h-100 text-center d-flex justify-content-between"> |     <div className="card p-3 h-100 text-center d-flex justify-content-between"> | ||||||
|  | |||||||
| @ -107,13 +107,13 @@ const CardViewDirectory = ({ | |||||||
|           {/* <li className="list-inline-item me-1 small"> |           {/* <li className="list-inline-item me-1 small"> | ||||||
|             <i className="fa-solid fa-briefcase me-2"></i> |             <i className="fa-solid fa-briefcase me-2"></i> | ||||||
|           </li> */} |           </li> */} | ||||||
|           <li className="list-inline-item text-break small ms-5"> |           <li className="list-inline-item text-break small px-1 ms-5"> | ||||||
|             {contact.organization} |             {contact.organization} | ||||||
|           </li> |           </li> | ||||||
|         </ul> |         </ul> | ||||||
|       </div> |       </div> | ||||||
|       <div |       <div | ||||||
|         className={`card-footer text-start px-1 py-1 ${IsActive && "cursor-pointer" |         className={`card-footer text-start px-9 py-1 ${IsActive && "cursor-pointer" | ||||||
|           }`} |           }`} | ||||||
|         onClick={() => { |         onClick={() => { | ||||||
|           if (IsActive) { |           if (IsActive) { | ||||||
| @ -123,6 +123,16 @@ const CardViewDirectory = ({ | |||||||
|         }} |         }} | ||||||
|       > |       > | ||||||
|         <hr className="my-0" /> |         <hr className="my-0" /> | ||||||
|  |         {contact.designation && ( | ||||||
|  |           <ul className="list-unstyled my-1 d-flex align-items-start  ms-2"> | ||||||
|  |             <li className="me-2"> | ||||||
|  |               <i class="fa-solid fa-id-badge ms-1"></i> | ||||||
|  |             </li> | ||||||
|  |             <li className="flex-grow-1 text-break small"> | ||||||
|  |               {contact.designation} | ||||||
|  |             </li> | ||||||
|  |           </ul> | ||||||
|  |         )} | ||||||
|         {contact.contactEmails[0] && ( |         {contact.contactEmails[0] && ( | ||||||
|           <ul className="list-unstyled my-1 d-flex align-items-start   ms-2"> |           <ul className="list-unstyled my-1 d-flex align-items-start   ms-2"> | ||||||
|             <li className="me-2"> |             <li className="me-2"> | ||||||
|  | |||||||
| @ -6,6 +6,7 @@ export const ContactSchema = z | |||||||
|     contactCategoryId: z.string().nullable().optional(), |     contactCategoryId: z.string().nullable().optional(), | ||||||
|     address: z.string().optional(), |     address: z.string().optional(), | ||||||
|     description: z.string().min(1, { message: "Description is required" }), |     description: z.string().min(1, { message: "Description is required" }), | ||||||
|  |     designation: z.string().min(1, {message:"Designation is requried"}), | ||||||
|     projectIds: z.array(z.string()).nullable().optional(), // min(1, "Project is required")
 |     projectIds: z.array(z.string()).nullable().optional(), // min(1, "Project is required")
 | ||||||
|     contactEmails: z |     contactEmails: z | ||||||
|       .array( |       .array( | ||||||
|  | |||||||
| @ -15,6 +15,10 @@ const ListViewDirectory = ({ | |||||||
| }) => { | }) => { | ||||||
|   const { dirActions, setDirActions } = useDir(); |   const { dirActions, setDirActions } = useDir(); | ||||||
| 
 | 
 | ||||||
|  |   // Get the first email and phone number if they exist | ||||||
|  |   const firstEmail = contact.contactEmails?.[0]; | ||||||
|  |   const firstPhone = contact.contactPhones?.[0]; | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <tr className={!IsActive ? "bg-light" : ""}> |     <tr className={!IsActive ? "bg-light" : ""}> | ||||||
|       <td |       <td | ||||||
| @ -47,36 +51,38 @@ const ListViewDirectory = ({ | |||||||
| 
 | 
 | ||||||
|       <td className="px-2" style={{ width: "20%" }}> |       <td className="px-2" style={{ width: "20%" }}> | ||||||
|         <div className="d-flex flex-column align-items-start text-truncate"> |         <div className="d-flex flex-column align-items-start text-truncate"> | ||||||
|          {contact.contactEmails.length > 0 ? (contact.contactEmails?.map((email, index) => ( |           {firstEmail ? ( | ||||||
|             <span key={email.id} className="text-truncate"> |             <span key={firstEmail.id} className="text-truncate"> | ||||||
|               <i |               <i | ||||||
|                 className={getEmailIcon(email.label)} |                 className={getEmailIcon(firstEmail.label)} | ||||||
|                 style={{ fontSize: "12px" }} |                 style={{ fontSize: "12px" }} | ||||||
|               ></i> |               ></i> | ||||||
|               <a |               <a | ||||||
|                 href={`mailto:${email.emailAddress}`} |                 href={`mailto:${firstEmail.emailAddress}`} | ||||||
|                 className="text-decoration-none ms-1" |                 className="text-decoration-none ms-1" | ||||||
|               > |               > | ||||||
|                 {email.emailAddress} |                 {firstEmail.emailAddress} | ||||||
|               </a> |               </a> | ||||||
|             </span> |             </span> | ||||||
|           ))):(<span className="small-text m-0 px-2">NA</span>)} |           ) : ( | ||||||
|  |             <span className="small-text m-0 px-2">NA</span> | ||||||
|  |           )} | ||||||
|         </div> |         </div> | ||||||
|       </td> |       </td> | ||||||
| 
 | 
 | ||||||
|       <td className="px-2" style={{ width: "20%" }}> |       <td className="px-2" style={{ width: "20%" }}> | ||||||
|         <div className="d-flex flex-column align-items-start text-truncate"> |         <div className="d-flex flex-column align-items-start text-truncate"> | ||||||
|            {contact.contactPhones?.length > 0 ? ( |           {firstPhone ? ( | ||||||
|             contact.contactPhones?.map((phone, index) => ( |             <span key={firstPhone.id}> | ||||||
|             <span key={phone.id}> |  | ||||||
|               <i |               <i | ||||||
|                 className={getPhoneIcon(phone.label)} |                 className={getPhoneIcon(firstPhone.label)} | ||||||
|                 style={{ fontSize: "12px" }} |                 style={{ fontSize: "12px" }} | ||||||
|               ></i> |               ></i> | ||||||
|               <span className="ms-1">{phone.phoneNumber}</span> |               <span className="ms-1">{firstPhone.phoneNumber}</span> | ||||||
|             </span> |             </span> | ||||||
|           )) |           ) : ( | ||||||
|           ):(<span className="text-small m-0 px-2">NA</span>)} |             <span className="text-small m-0 px-2">NA</span> | ||||||
|  |           )} | ||||||
|         </div> |         </div> | ||||||
|       </td> |       </td> | ||||||
| 
 | 
 | ||||||
| @ -88,12 +94,6 @@ const ListViewDirectory = ({ | |||||||
|         {contact.organization} |         {contact.organization} | ||||||
|       </td> |       </td> | ||||||
| 
 | 
 | ||||||
|       {/* <td className="px-2" style={{ width: "10%" }}> |  | ||||||
|         <span className="badge badge-outline-secondary"> |  | ||||||
|           {contact?.contactCategory?.name || "Other"} |  | ||||||
|         </span> |  | ||||||
|       </td> */} |  | ||||||
|        |  | ||||||
|       <td className="px-2" style={{ width: "10%" }}> |       <td className="px-2" style={{ width: "10%" }}> | ||||||
|         <span className="text-truncate"> |         <span className="text-truncate"> | ||||||
|           {contact?.contactCategory?.name || "Other"} |           {contact?.contactCategory?.name || "Other"} | ||||||
| @ -119,7 +119,8 @@ const ListViewDirectory = ({ | |||||||
|         {!IsActive && ( |         {!IsActive && ( | ||||||
|           <i |           <i | ||||||
|             className={`bx ${ |             className={`bx ${ | ||||||
|               dirActions.action && dirActions.id === contact.id ?  "bx-loader-alt bx-spin" |               dirActions.action && dirActions.id === contact.id | ||||||
|  |                 ? "bx-loader-alt bx-spin" | ||||||
|                 : "bx-recycle" |                 : "bx-recycle" | ||||||
|             } me-1 text-primary cursor-pointer`} |             } me-1 text-primary cursor-pointer`} | ||||||
|             title="Restore" |             title="Restore" | ||||||
|  | |||||||
| @ -14,7 +14,11 @@ import useMaster, { | |||||||
| } from "../../hooks/masterHook/useMaster"; | } from "../../hooks/masterHook/useMaster"; | ||||||
| import { useDispatch, useSelector } from "react-redux"; | import { useDispatch, useSelector } from "react-redux"; | ||||||
| import { changeMaster } from "../../slices/localVariablesSlice"; | import { changeMaster } from "../../slices/localVariablesSlice"; | ||||||
| import { useBuckets, useOrganization } from "../../hooks/useDirectory"; | import { | ||||||
|  |   useBuckets, | ||||||
|  |   useDesignation, | ||||||
|  |   useOrganization, | ||||||
|  | } from "../../hooks/useDirectory"; | ||||||
| import { useProjects } from "../../hooks/useProjects"; | import { useProjects } from "../../hooks/useProjects"; | ||||||
| import SelectMultiple from "../common/SelectMultiple"; | import SelectMultiple from "../common/SelectMultiple"; | ||||||
| import { ContactSchema } from "./DirectorySchema"; | import { ContactSchema } from "./DirectorySchema"; | ||||||
| @ -33,8 +37,11 @@ const ManageDirectory = ({ submitContact, onCLosed }) => { | |||||||
|   const { contactCategory, loading: contactCategoryLoading } = |   const { contactCategory, loading: contactCategoryLoading } = | ||||||
|     useContactCategory(); |     useContactCategory(); | ||||||
|   const { organizationList, loading: orgLoading } = useOrganization(); |   const { organizationList, loading: orgLoading } = useOrganization(); | ||||||
|  |   const { designationList, loading: designloading } = useDesignation(); | ||||||
|   const { contactTags, loading: Tagloading } = useContactTags(); |   const { contactTags, loading: Tagloading } = useContactTags(); | ||||||
|   const [IsSubmitting, setSubmitting] = useState(false); |   const [IsSubmitting, setSubmitting] = useState(false); | ||||||
|  |   const [showSuggestions,setShowSuggestions] = useState(false); | ||||||
|  |   const [filteredDesignationList, setFilteredDesignationList] = useState([]); | ||||||
|   const dispatch = useDispatch(); |   const dispatch = useDispatch(); | ||||||
| 
 | 
 | ||||||
|   const methods = useForm({ |   const methods = useForm({ | ||||||
| @ -45,6 +52,7 @@ const ManageDirectory = ({ submitContact, onCLosed }) => { | |||||||
|       contactCategoryId: null, |       contactCategoryId: null, | ||||||
|       address: "", |       address: "", | ||||||
|       description: "", |       description: "", | ||||||
|  |       designation: "", | ||||||
|       projectIds: [], |       projectIds: [], | ||||||
|       contactEmails: [], |       contactEmails: [], | ||||||
|       contactPhones: [], |       contactPhones: [], | ||||||
| @ -106,6 +114,25 @@ const ManageDirectory = ({ submitContact, onCLosed }) => { | |||||||
| 
 | 
 | ||||||
|   const watchBucketIds = watch("bucketIds"); |   const watchBucketIds = watch("bucketIds"); | ||||||
| 
 | 
 | ||||||
|  |   // handle logic when input of desgination is changed | ||||||
|  |   const handleDesignationChange = (e) => { | ||||||
|  |     const val = e.target.value; | ||||||
|  | 
 | ||||||
|  |     const matches = designationList.filter((org) => | ||||||
|  |       org.toLowerCase().includes(val.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |     setFilteredDesignationList(matches); | ||||||
|  |     setShowSuggestions(true); | ||||||
|  |     setTimeout(() => setShowSuggestions(false), 5000); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   // handle logic when designation is selected | ||||||
|  |   const handleSelectDesignation = (val) => { | ||||||
|  |     setShowSuggestions(false); | ||||||
|  |     setValue("designation", val); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|   const toggleBucketId = (id) => { |   const toggleBucketId = (id) => { | ||||||
|     const updated = watchBucketIds?.includes(id) |     const updated = watchBucketIds?.includes(id) | ||||||
|       ? watchBucketIds.filter((val) => val !== id) |       ? watchBucketIds.filter((val) => val !== id) | ||||||
| @ -168,6 +195,55 @@ const ManageDirectory = ({ submitContact, onCLosed }) => { | |||||||
|             /> |             /> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |         <div className="row mt-1"> | ||||||
|  |           <div className="col-md-6  text-start"> | ||||||
|  |             <label className="form-label">Designation</label> | ||||||
|  |             <input | ||||||
|  |               className="form-control form-control-sm" | ||||||
|  |               {...register("designation")} | ||||||
|  |               onChange={handleDesignationChange} | ||||||
|  |             /> | ||||||
|  |             {showSuggestions && filteredDesignationList.length > 0 && ( | ||||||
|  |               <ul | ||||||
|  |                 className="list-group shadow-sm position-absolute bg-white border w-50 zindex-tooltip" | ||||||
|  |                 style={{ | ||||||
|  |                   maxHeight: "180px", | ||||||
|  |                   overflowY: "auto", | ||||||
|  |                   marginTop: "2px", | ||||||
|  |                   zIndex: 1000, | ||||||
|  |                   borderRadius: "0px", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 {filteredDesignationList.map((designation) => ( | ||||||
|  |                   <li | ||||||
|  |                     key={designation} | ||||||
|  |                     className="list-group-item list-group-item-action border-none " | ||||||
|  |                     style={{ | ||||||
|  |                       cursor: "pointer", | ||||||
|  |                       padding: "5px 12px", | ||||||
|  |                       fontSize: "14px", | ||||||
|  |                       transition: "background-color 0.2s", | ||||||
|  |                     }} | ||||||
|  |                     onMouseDown={() => handleSelectDesignation(designation)} | ||||||
|  |                     onMouseEnter={(e) => | ||||||
|  |                       (e.currentTarget.style.backgroundColor = "#f8f9fa") | ||||||
|  |                     } | ||||||
|  |                     onMouseLeave={(e) => | ||||||
|  |                       (e.currentTarget.style.backgroundColor = "transparent") | ||||||
|  |                     } | ||||||
|  |                   > | ||||||
|  |                     {designation} | ||||||
|  |                   </li> | ||||||
|  |                 ))} | ||||||
|  |               </ul> | ||||||
|  |             )} | ||||||
|  |             {errors.designation && ( | ||||||
|  |               <small className="danger-text"> | ||||||
|  |                 {errors.designation.message} | ||||||
|  |               </small> | ||||||
|  |             )} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|         <div className="row mt-1"> |         <div className="row mt-1"> | ||||||
|           <div className="col-md-6"> |           <div className="col-md-6"> | ||||||
|             {emailFields.map((field, index) => ( |             {emailFields.map((field, index) => ( | ||||||
| @ -381,7 +457,6 @@ const ManageDirectory = ({ submitContact, onCLosed }) => { | |||||||
|                   </div> |                   </div> | ||||||
|                 </li> |                 </li> | ||||||
|               ))} |               ))} | ||||||
|              |  | ||||||
|             </ul> |             </ul> | ||||||
|             {errors.bucketIds && ( |             {errors.bucketIds && ( | ||||||
|               <small className="danger-text mt-0"> |               <small className="danger-text mt-0"> | ||||||
|  | |||||||
| @ -153,7 +153,7 @@ const NoteCardDirectoryEditable = ({ | |||||||
|                       .utc(noteItem?.createdAt) |                       .utc(noteItem?.createdAt) | ||||||
|                       .add(5, "hours") |                       .add(5, "hours") | ||||||
|                       .add(30, "minutes") |                       .add(30, "minutes") | ||||||
|                       .format("MMMM DD, YYYY [at] hh:mm A")} |                       .format("DD MMMM, YYYY [at] hh:mm A")} | ||||||
|                   </span> |                   </span> | ||||||
|                 </span> |                 </span> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4,7 +4,6 @@ import Avatar from "../common/Avatar"; | |||||||
| import { useForm } from "react-hook-form"; | import { useForm } from "react-hook-form"; | ||||||
| import { z } from "zod"; | import { z } from "zod"; | ||||||
| import { zodResolver } from "@hookform/resolvers/zod"; | import { zodResolver } from "@hookform/resolvers/zod"; | ||||||
| import { showText } from "pdf-lib"; |  | ||||||
| import { DirectoryRepository } from "../../repositories/DirectoryRepository"; | import { DirectoryRepository } from "../../repositories/DirectoryRepository"; | ||||||
| import moment from "moment"; | import moment from "moment"; | ||||||
| import { cacheData, getCachedData } from "../../slices/apiDataManager"; | import { cacheData, getCachedData } from "../../slices/apiDataManager"; | ||||||
| @ -19,15 +18,17 @@ const schema = z.object({ | |||||||
| const NotesDirectory = ({ | const NotesDirectory = ({ | ||||||
|   refetchProfile, |   refetchProfile, | ||||||
|   isLoading, |   isLoading, | ||||||
|   contactProfile, |   contactProfile, // This contactProfile now reliably includes firstName, middleName, lastName, and fullName | ||||||
|   setProfileContact, |   setProfileContact, | ||||||
| }) => { | }) => { | ||||||
|   const [IsActive, setIsActive] = useState(true); |   const [IsActive, setIsActive] = useState(true); | ||||||
|   const { contactNotes, refetch } = useContactNotes(contactProfile?.id, true); |   const { contactNotes, refetch } = useContactNotes( | ||||||
|  |     contactProfile?.id, | ||||||
|  |     IsActive | ||||||
|  |   ); | ||||||
| 
 | 
 | ||||||
|   const [NotesData, setNotesData] = useState(); |  | ||||||
|   const [IsSubmitting, setIsSubmitting] = useState(false); |   const [IsSubmitting, setIsSubmitting] = useState(false); | ||||||
|   const [addNote, setAddNote] = useState(true); |   const [showEditor, setShowEditor] = useState(false); | ||||||
|   const { |   const { | ||||||
|     register, |     register, | ||||||
|     handleSubmit, |     handleSubmit, | ||||||
| @ -67,61 +68,78 @@ const NotesDirectory = ({ | |||||||
|       ) { |       ) { | ||||||
|         const updatedProfile = { |         const updatedProfile = { | ||||||
|           ...cached_contactProfile.data, |           ...cached_contactProfile.data, | ||||||
|           notes: [...(cached_contactProfile.notes || []), createdNote], |           notes: [...(cached_contactProfile.data.notes || []), createdNote], | ||||||
|         }; |         }; | ||||||
|         cacheData("Contact Profile", updatedProfile); |         cacheData("Contact Profile", { | ||||||
|  |           contactId: contactProfile?.id, | ||||||
|  |           data: updatedProfile, | ||||||
|  |         }); | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       setValue("note", ""); |       setValue("note", ""); | ||||||
|       setIsSubmitting(false); |       setIsSubmitting(false); | ||||||
|       showToast("Note added successfully!", "success"); |       showToast("Note added successfully!", "success"); | ||||||
|       setAddNote(true); |       setShowEditor(false); | ||||||
|       setIsActive(true); |       setIsActive(true); | ||||||
|  |       refetch(contactProfile?.id, true); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       setIsSubmitting(false); |       setIsSubmitting(false); | ||||||
|       const msg = |       const msg = | ||||||
|         error.response.data.message || |         error.response?.data?.message || | ||||||
|         error.message || |         error.message || | ||||||
|         "Error occured during API calling"; |         "Error occurred during API calling"; | ||||||
|       showToast(msg, "error"); |       showToast(msg, "error"); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const onCancel = () => { |   const onCancel = () => { | ||||||
|     setValue("note", ""); |     setValue("note", ""); | ||||||
|  |     setShowEditor(false); | ||||||
|  |   }; | ||||||
| 
 | 
 | ||||||
|   }; |  | ||||||
|   const handleSwitch = () => { |   const handleSwitch = () => { | ||||||
|     setIsActive(!IsActive); |     setIsActive((prevIsActive) => { | ||||||
|     if (IsActive) { |       const newState = !prevIsActive; | ||||||
|       refetch(contactProfile?.id, false); |       refetch(contactProfile?.id, newState); | ||||||
|     } |       return newState; | ||||||
|  |     }); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   // Use the fullName from contactProfile, which now includes middle and last names if available | ||||||
|  |   const contactName = | ||||||
|  |     contactProfile?.fullName || contactProfile?.firstName || "Contact"; | ||||||
|  |   const noNotesMessage = `Be the first to share your insights! ${contactName} currently has no notes.`; | ||||||
|  | 
 | ||||||
|  |   const notesToDisplay = IsActive | ||||||
|  |     ? contactProfile?.notes || [] | ||||||
|  |     : contactNotes || []; | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className="text-start"> |     <div className="text-start mt-10"> | ||||||
|       <div className="d-flex align-items-center justify-content-between"> |       <div className="d-flex align-items-center justify-content-between"> | ||||||
|         <p className="fw-semibold m-0">Notes :</p> |         <div className="row w-100 align-items-center"> | ||||||
|  |           <div className="col col-2"> | ||||||
|  |             <p className="fw-semibold m-0 ms-3">Notes :</p> | ||||||
|           </div> |           </div> | ||||||
|       <div className="d-flex align-items-center justify-content-between mb-5"> |           <div className="col d-flex justify-content-end gap-2 pe-0"> | ||||||
|   <div className="m-0 d-flex align-items-center"> |             {" "} | ||||||
|     {contactNotes?.length > 0 ? ( |             <div className="d-flex align-items-center justify-content-between"> | ||||||
|       <label className="switch switch-primary"> |               <label | ||||||
|  |                 className="switch switch-primary" | ||||||
|  |                 style={{ | ||||||
|  |                   visibility: | ||||||
|  |                     contactProfile?.notes?.length > 0 || | ||||||
|  |                     contactNotes?.length > 0 | ||||||
|  |                       ? "visible" | ||||||
|  |                       : "hidden", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|               <input |               <input | ||||||
|                   type="checkbox" |                   type="checkbox" | ||||||
|                   className="switch-input" |                   className="switch-input" | ||||||
|                   onChange={() => handleSwitch(!IsActive)} |                   onChange={() => handleSwitch(!IsActive)} | ||||||
|                   value={IsActive} |                   value={IsActive} | ||||||
|                 /> |                 /> | ||||||
|         <span className="switch-toggle-slider"> |  | ||||||
|           <span className="switch-on"></span> |  | ||||||
|           <span className="switch-off"></span> |  | ||||||
|         </span> |  | ||||||
|         <span className="switch-label">Include Deleted Notes</span> |  | ||||||
|       </label> |  | ||||||
|     ) : ( |  | ||||||
|       <div style={{ visibility: "hidden" }}> |  | ||||||
|         <label className="switch switch-primary"> |  | ||||||
|                 <input type="checkbox" className="switch-input" /> |                 <input type="checkbox" className="switch-input" /> | ||||||
|                 <span className="switch-toggle-slider"> |                 <span className="switch-toggle-slider"> | ||||||
|                   <span className="switch-on"></span> |                   <span className="switch-on"></span> | ||||||
| @ -129,40 +147,43 @@ const NotesDirectory = ({ | |||||||
|                 </span> |                 </span> | ||||||
|                 <span className="switch-label">Include Deleted Notes</span> |                 <span className="switch-label">Include Deleted Notes</span> | ||||||
|               </label> |               </label> | ||||||
|  | 
 | ||||||
|  |               {!showEditor && ( | ||||||
|  |                 <div className="d-flex justify-content-end"> | ||||||
|  |                   <button | ||||||
|  |                     type="button" | ||||||
|  |                     className="btn btn-sm d-flex align-items-center" | ||||||
|  |                     onClick={() => setShowEditor(true)} | ||||||
|  |                     style={{ | ||||||
|  |                       color: "#6c757d", | ||||||
|  |                       backgroundColor: "transparent", | ||||||
|  |                       boxShadow: "none", | ||||||
|  |                       border: "none", | ||||||
|  |                     }} | ||||||
|  |                   > | ||||||
|  |                     <i | ||||||
|  |                       className="bx bx-plus-circle me-0 text-primary" | ||||||
|  |                       style={{ fontSize: "1.5rem", color: "#6c757d" }} | ||||||
|  |                     ></i> | ||||||
|  |                     Add a Note | ||||||
|  |                   </button> | ||||||
|                 </div> |                 </div> | ||||||
|               )} |               )} | ||||||
|             </div> |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
| 
 | 
 | ||||||
|   <div className="d-flex justify-content-end"> |       {showEditor && ( | ||||||
|  |         <div className="card m-2 mb-5 position-relative"> | ||||||
|           <span |           <span | ||||||
|        className={`btn btn-sm ${addNote ? "btn-secondary" : "btn-primary"}`} |  | ||||||
|       onClick={() => setAddNote(!addNote)} |  | ||||||
|     > |  | ||||||
|       {addNote ? "Hide Editor" : "Add a Note"} |  | ||||||
|     </span> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|       {addNote && ( |  | ||||||
|         <div className="card m-2 mb-5"> |  | ||||||
|           <button |  | ||||||
|             type="button" |             type="button" | ||||||
|             class="btn  btn-close btn-secondary position-absolute top-0 end-0 m-2 mt-3 rounded-circle" |             className="position-absolute top-0 end-0  mt-3 bg-secondary rounded-circle" | ||||||
|             aria-label="Close" |             aria-label="Close" | ||||||
|             style={{ backgroundColor: "#eee", color: "white" }} |             onClick={() => setShowEditor(false)} | ||||||
|             onClick={() => setAddNote(!addNote)} |  | ||||||
|           ></button> |  | ||||||
|           {/* <div className="d-flex justify-content-end px-2"> |  | ||||||
|             <span |  | ||||||
|               className={`btn btn-sm   ${ |  | ||||||
|                 addNote ? "btn-danger" : "btn-primary" |  | ||||||
|               }`} |  | ||||||
|               onClick={() => setAddNote(!addNote)} |  | ||||||
|           > |           > | ||||||
|               {addNote ? "Hide Editor" : "Add Note"} |             <i className="bx bx-x fs-5  p-1 text-white"></i> | ||||||
|           </span> |           </span> | ||||||
|           </div> */} |  | ||||||
|           <form onSubmit={handleSubmit(onSubmit)}> |           <form onSubmit={handleSubmit(onSubmit)}> | ||||||
|             <Editor |             <Editor | ||||||
|               value={noteValue} |               value={noteValue} | ||||||
| @ -171,7 +192,7 @@ const NotesDirectory = ({ | |||||||
|               onCancel={onCancel} |               onCancel={onCancel} | ||||||
|               onSubmit={handleSubmit(onSubmit)} |               onSubmit={handleSubmit(onSubmit)} | ||||||
|             /> |             /> | ||||||
|             {errors.notes && ( |             {errors.note && ( | ||||||
|               <p className="text-danger small mt-1">{errors.note.message}</p> |               <p className="text-danger small mt-1">{errors.note.message}</p> | ||||||
|             )} |             )} | ||||||
|           </form> |           </form> | ||||||
| @ -185,8 +206,9 @@ const NotesDirectory = ({ | |||||||
|             <p>Loading...</p>{" "} |             <p>Loading...</p>{" "} | ||||||
|           </div> |           </div> | ||||||
|         )} |         )} | ||||||
|         {!isLoading && |         {!isLoading && notesToDisplay.length > 0 | ||||||
|           [...(IsActive ? contactProfile?.notes || [] : contactNotes || [])] |           ? notesToDisplay | ||||||
|  |               .slice() | ||||||
|               .reverse() |               .reverse() | ||||||
|               .map((noteItem) => ( |               .map((noteItem) => ( | ||||||
|                 <NoteCardDirectory |                 <NoteCardDirectory | ||||||
| @ -198,21 +220,10 @@ const NotesDirectory = ({ | |||||||
|                   setProfileContact={setProfileContact} |                   setProfileContact={setProfileContact} | ||||||
|                   key={noteItem.id} |                   key={noteItem.id} | ||||||
|                 /> |                 /> | ||||||
|             ))} |               )) | ||||||
| 
 |           : !isLoading && | ||||||
|         {IsActive && ( |             !showEditor && ( | ||||||
|           <div> |               <div className="text-center mt-5">{noNotesMessage}</div> | ||||||
|             {!isLoading && contactProfile?.notes.length == 0 && !addNote && ( |  | ||||||
|               <div className="text-center mt-5">No Notes Found</div> |  | ||||||
|             )} |  | ||||||
|           </div> |  | ||||||
|         )} |  | ||||||
|         {!IsActive && ( |  | ||||||
|           <div> |  | ||||||
|             {!isLoading && contactNotes.length == 0 && !addNote && ( |  | ||||||
|               <div className="text-center  mt-5">No Notes Found</div> |  | ||||||
|             )} |  | ||||||
|           </div> |  | ||||||
|             )} |             )} | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -8,9 +8,10 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|   const { contactProfile, loading, refetch } = useContactProfile(contact?.id); |   const { contactProfile, loading, refetch } = useContactProfile(contact?.id); | ||||||
|   const [copiedIndex, setCopiedIndex] = useState(null); |   const [copiedIndex, setCopiedIndex] = useState(null); | ||||||
| 
 | 
 | ||||||
|   const [profileContact, setProfileContact] = useState(); |   const [profileContactState, setProfileContactState] = useState(null); | ||||||
|   const [expanded, setExpanded] = useState(false); |   const [expanded, setExpanded] = useState(false); | ||||||
|   const description = contactProfile?.description || ""; | 
 | ||||||
|  |   const description = profileContactState?.description || ""; | ||||||
|   const limit = 500; |   const limit = 500; | ||||||
| 
 | 
 | ||||||
|   const toggleReadMore = () => setExpanded(!expanded); |   const toggleReadMore = () => setExpanded(!expanded); | ||||||
| @ -19,14 +20,51 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|   const displayText = expanded |   const displayText = expanded | ||||||
|     ? description |     ? description | ||||||
|     : description.slice(0, limit) + (isLong ? "..." : ""); |     : description.slice(0, limit) + (isLong ? "..." : ""); | ||||||
|  | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     setProfileContact(contactProfile); |     if (contactProfile) { | ||||||
|   }, [contactProfile]); |       const names = (contact?.name || "").trim().split(" "); | ||||||
|  |       let firstName = ""; | ||||||
|  |       let middleName = ""; | ||||||
|  |       let lastName = ""; | ||||||
|  |       let fullName = contact?.name || ""; | ||||||
|  | 
 | ||||||
|  |       // Logic to determine first, middle, and last names | ||||||
|  |       if (names.length === 1) { | ||||||
|  |         firstName = names[0]; | ||||||
|  |       } else if (names.length === 2) { | ||||||
|  |         firstName = names[0]; | ||||||
|  |         lastName = names[1]; | ||||||
|  |       } else if (names.length >= 3) { | ||||||
|  |         firstName = names[0]; | ||||||
|  |         middleName = names[1]; // This was an error in the original prompt, corrected to names[1] | ||||||
|  |         lastName = names[names.length - 1]; | ||||||
|  |         // Reconstruct full name to be precise with spacing | ||||||
|  |         fullName = `${firstName} ${middleName ? middleName + ' ' : ''}${lastName}`; | ||||||
|  |       } else { | ||||||
|  |         // Fallback if no names or empty string | ||||||
|  |         firstName = "Contact"; | ||||||
|  |         fullName = "Contact"; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |       setProfileContactState({ | ||||||
|  |         ...contactProfile, | ||||||
|  |         firstName: contactProfile.firstName || firstName, | ||||||
|  |         // Adding middleName and lastName to the state for potential future use or more granular access | ||||||
|  |         middleName: contactProfile.middleName || middleName, | ||||||
|  |         lastName: contactProfile.lastName || lastName, | ||||||
|  |         fullName: contactProfile.fullName || fullName, // Prioritize fetched fullName, fallback to derived | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }, [contactProfile, contact?.name]); | ||||||
|  | 
 | ||||||
|   const handleCopy = (email, index) => { |   const handleCopy = (email, index) => { | ||||||
|     navigator.clipboard.writeText(email); |     navigator.clipboard.writeText(email); | ||||||
|     setCopiedIndex(index); |     setCopiedIndex(index); | ||||||
|     setTimeout(() => setCopiedIndex(null), 2000); // Reset after 2 seconds |     setTimeout(() => setCopiedIndex(null), 2000); | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className="p-1"> |     <div className="p-1"> | ||||||
|       <div className="text-center m-0 p-0"> |       <div className="text-center m-0 p-0"> | ||||||
| @ -47,30 +85,34 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|           <div className="d-flex flex-column text-start ms-1"> |           <div className="d-flex flex-column text-start ms-1"> | ||||||
|             <span className="m-0 fw-semibold">{contact?.name}</span> |             <span className="m-0 fw-semibold">{contact?.name}</span> | ||||||
|             <small className="text-secondary small-text"> |             <small className="text-secondary small-text"> | ||||||
|               {contactProfile?.tags?.map((tag) => tag.name).join(" | ")} |               {profileContactState?.designation} | ||||||
|             </small> |             </small> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div className="row"> |         <div className="row ms-9"> | ||||||
|           <div className="col-12 col-md-6 d-flex flex-column text-start"> |           <div className="col-12 col-md-6 d-flex flex-column text-start"> | ||||||
|             {contactProfile?.contactEmails?.length > 0 && ( |             {profileContactState?.contactEmails?.length > 0 && ( | ||||||
|               <div className="d-flex mb-2"> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |                 <div | ||||||
|                   <p className="m-0">Email:</p> |                   className="d-flex align-items-start" | ||||||
|  |                   style={{ width: "100px", minWidth: "130px" }} | ||||||
|  |                 > | ||||||
|  |                   <span className="d-flex"> | ||||||
|  |                     <i className="bx bx-envelope bx-xs me-2 mt-1"></i> | ||||||
|  |                     <span>Email</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "45px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div style={{ flex: 1 }}> |                 <div style={{ flex: 1 }}> | ||||||
|                   <ul className="list-unstyled mb-0"> |                   <ul className="list-unstyled mb-0"> | ||||||
|                     {contactProfile.contactEmails.map((email, idx) => ( |                     {profileContactState.contactEmails.map((email, idx) => ( | ||||||
|                       <li className="d-flex align-items-center mb-1" key={idx}> |                       <li className="d-flex align-items-center mb-1" key={idx}> | ||||||
|                         <i className="bx bx-envelope bx-xs me-1 mt-1"></i> |                         <span className="me-1 text-break overflow-wrap"> | ||||||
|                         <span className="me-1 flex-grow text-break overflow-wrap"> |  | ||||||
|                           {email.emailAddress} |                           {email.emailAddress} | ||||||
|                         </span> |                         </span> | ||||||
|                         <i |                         <i | ||||||
|                           className={`bx bx-copy-alt cursor-pointer bx-xs text-start ${ |                           className={`bx bx-copy-alt cursor-pointer bx-xs text-start ${copiedIndex === idx ? "text-secondary" : "text-primary" | ||||||
|                             copiedIndex === idx |  | ||||||
|                               ? "text-secondary" |  | ||||||
|                               : "text-primary" |  | ||||||
|                             }`} |                             }`} | ||||||
|                           title={copiedIndex === idx ? "Copied!" : "Copy Email"} |                           title={copiedIndex === idx ? "Copied!" : "Copy Email"} | ||||||
|                           style={{ flexShrink: 0 }} |                           style={{ flexShrink: 0 }} | ||||||
| @ -83,17 +125,22 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
| 
 | 
 | ||||||
|             {contactProfile?.contactPhones?.length > 0 && ( |             {profileContactState?.contactPhones?.length > 0 && ( | ||||||
|               <div className="d-flex mb-2"> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|                   <p className="m-0">Phone : </p> |                   <span className="d-flex align-items-center"> | ||||||
|  |                     <i className="bx bx-phone bx-xs me-2"></i> | ||||||
|  |                     <span>Phone</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "40px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div> |                 <div> | ||||||
|                   <ul className="list-inline mb-0"> |                   <ul className="list-inline mb-0"> | ||||||
|                     {contactProfile?.contactPhones.map((phone, idx) => ( |                     {profileContactState.contactPhones.map((phone, idx) => ( | ||||||
|                       <li className="list-inline-item me-3" key={idx}> |                       <li className="list-inline-item me-1" key={idx}> | ||||||
|                         <i className="bx bx-phone bx-xs me-1"></i> |  | ||||||
|                         {phone.phoneNumber} |                         {phone.phoneNumber} | ||||||
|  |                         {idx < profileContactState.contactPhones.length - 1 && ","} | ||||||
|                       </li> |                       </li> | ||||||
|                     ))} |                     ))} | ||||||
|                   </ul> |                   </ul> | ||||||
| @ -101,74 +148,93 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
| 
 | 
 | ||||||
|             {contactProfile?.createdAt && ( |             {profileContactState?.createdAt && ( | ||||||
|               <div className="d-flex  mb-2"> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|                   <p className="m-0">Created : </p> |                   <span className="d-flex align-items-center"> | ||||||
|  |                     <i className="bx bx-calendar-week bx-xs me-2"></i> | ||||||
|  |                     <span>Created</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "30px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div className="d-flex align-items-center"> |                 <div className="d-flex align-items-center"> | ||||||
|                   <li className="list-inline-item"> |                   <span> | ||||||
|                     <i className="bx bx-calendar-week bx-xs me-1"></i> |                     {moment(profileContactState.createdAt).format("DD MMMM, YYYY")} | ||||||
|                     {moment(contactProfile.createdAt).format("MMMM, DD YYYY")} |                   </span> | ||||||
|                   </li> |  | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|             {contactProfile?.address && ( | 
 | ||||||
|               <div className="d-flex mb-2"> |             {profileContactState?.address && ( | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                   <p className="m-0">Location:</p> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|                 </div> |                   <span className="d-flex align-items-start"> | ||||||
|                 <div className="d-flex align-items-center"> |                     <i className="bx bx-map bx-xs me-2 mt-1"></i> | ||||||
|                   <i className="bx bx-map bx-xs me-1 "></i> |                     <span>Location</span> | ||||||
|                   <span className="text-break small"> |  | ||||||
|                     {contactProfile.address} |  | ||||||
|                   </span> |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "26px" }}>:</span> | ||||||
|  |                 </div> | ||||||
|  |                 <div> | ||||||
|  |                   <span className="text-break small">{profileContactState.address}</span> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div className="col-12 col-md-6 d-flex flex-column text-start"> |           <div className="col-12 col-md-6 d-flex flex-column text-start"> | ||||||
|             {contactProfile?.organization && ( |             {profileContactState?.organization && ( | ||||||
|               <div className="d-flex mb-2"> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|                   <p className="m-0">Orgnization : </p> |                   <span className="d-flex align-items-center"> | ||||||
|                 </div> |  | ||||||
|                 <div className="d-flex align-items-center"> |  | ||||||
|                     <i className="fa-solid fa-briefcase me-2"></i> |                     <i className="fa-solid fa-briefcase me-2"></i> | ||||||
|  |                     <span>Organization</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span className="ms-2">:</span> | ||||||
|  |                 </div> | ||||||
| 
 | 
 | ||||||
|  |                 <div className="d-flex align-items-center"> | ||||||
|                   <span style={{ wordBreak: "break-word" }}> |                   <span style={{ wordBreak: "break-word" }}> | ||||||
|                     {contactProfile.organization} |                     {profileContactState.organization} | ||||||
|                   </span> |                   </span> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|             {contactProfile?.contactCategory && ( | 
 | ||||||
|               <div className="d-flex mb-2"> |             {profileContactState?.contactCategory && ( | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                   <p className="m-0">Category : </p> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|  |                   <span className="d-flex align-items-center"> | ||||||
|  |                     <i className="bx bx-user bx-xs me-2"></i> | ||||||
|  |                     <span>Category</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "28px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div> |                 <div> | ||||||
|                   <ul className="list-inline mb-0"> |                   <ul className="list-inline mb-0"> | ||||||
|                     <li className="list-inline-item"> |                     <li className="list-inline-item"> | ||||||
|                       <i className="bx bx-user bx-xs me-1"></i> |                       {profileContactState.contactCategory.name} | ||||||
|                       {contactProfile.contactCategory.name} |  | ||||||
|                     </li> |                     </li> | ||||||
|                   </ul> |                   </ul> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|              {contactProfile?.tags?.length > 0 && ( | 
 | ||||||
|               <div className="d-flex mb-2"> |             {profileContactState?.tags?.length > 0 && ( | ||||||
|                 <div style={{ width: "100px", minWidth: "100px" }}> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                   <p className="m-0">Tags : </p> |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|  |                   <span className="d-flex align-items-center"> | ||||||
|  |                     <i className="fa-solid fa-tag me-2"></i> | ||||||
|  |                     <span>Tags</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "60px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div> |                 <div> | ||||||
|                   <ul className="list-inline mb-0"> |                   <ul className="list-inline mb-0"> | ||||||
|                     {contactProfile.tags.map((tag, index) => ( |                     {profileContactState.tags.map((tag, index) => ( | ||||||
|                       <li key={index} className="list-inline-item"> |                       <li key={index} className="list-inline-item"> | ||||||
|                         <i className="fa-solid fa-tag  me-1"></i> |  | ||||||
|                         {tag.name} |                         {tag.name} | ||||||
|                       </li> |                       </li> | ||||||
|                     ))} |                     ))} | ||||||
| @ -177,16 +243,19 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
| 
 | 
 | ||||||
|             {contactProfile?.buckets?.length > 0 && ( |             {profileContactState?.buckets?.length > 0 && ( | ||||||
|               <div className="d-flex "> |               <div className="d-flex mb-2 align-items-start"> | ||||||
|                 {contactProfile?.contactEmails?.length > 0 && ( |                 <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|                   <div className="d-flex mb-2 align-items-center"> |                   <span className="d-flex align-items-center"> | ||||||
|                     <div style={{ width: "100px", minWidth: "100px" }}> |                     <i className="bx bx-layer me-1"></i> | ||||||
|                       <p className="m-0">Buckets : </p> |                     <span>Buckets</span> | ||||||
|  |                   </span> | ||||||
|  |                   <span style={{ marginLeft: "35px" }}>:</span> | ||||||
|                 </div> |                 </div> | ||||||
|  | 
 | ||||||
|                 <div> |                 <div> | ||||||
|                   <ul className="list-inline mb-0"> |                   <ul className="list-inline mb-0"> | ||||||
|                         {contactProfile.buckets.map((bucket) => ( |                     {profileContactState.buckets.map((bucket) => ( | ||||||
|                       <li className="list-inline-item me-2" key={bucket.id}> |                       <li className="list-inline-item me-2" key={bucket.id}> | ||||||
|                         <span className="badge bg-label-primary my-1"> |                         <span className="badge bg-label-primary my-1"> | ||||||
|                           {bucket.name} |                           {bucket.name} | ||||||
| @ -198,50 +267,63 @@ const ProfileContactDirectory = ({ contact, setOpen_contact, closeModal }) => { | |||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
|             )} | 
 | ||||||
|           </div> |           {profileContactState?.projects?.length > 0 && ( | ||||||
|         </div> |  | ||||||
|         {contactProfile?.projects?.length > 0 && ( |  | ||||||
|             <div className="d-flex mb-2 align-items-start"> |             <div className="d-flex mb-2 align-items-start"> | ||||||
|             <div style={{ minWidth: "100px" }}> |               <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|               <p className="m-0 text-start">Projects :</p> |                 <span className="d-flex align-items-center"> | ||||||
|  |                   <i className="bx bx-building-house me-1"></i> | ||||||
|  |                   <span>Projects</span> | ||||||
|  |                 </span> | ||||||
|  |                 <span style={{ marginLeft: "28px" }}>:</span> | ||||||
|               </div> |               </div> | ||||||
|  | 
 | ||||||
|               <div className="text-start"> |               <div className="text-start"> | ||||||
|                 <ul className="list-inline mb-0"> |                 <ul className="list-inline mb-0"> | ||||||
|                 {contactProfile.projects.map((project, index) => ( |                   {profileContactState.projects.map((project, index) => ( | ||||||
|                     <li className="list-inline-item me-2" key={project.id}> |                     <li className="list-inline-item me-2" key={project.id}> | ||||||
|                       {project.name} |                       {project.name} | ||||||
|                     {index < contactProfile.projects.length - 1 && ","} |                       {index < profileContactState.projects.length - 1 && ","} | ||||||
|                     </li> |                     </li> | ||||||
|                   ))} |                   ))} | ||||||
|                 </ul> |                 </ul> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           )} |           )} | ||||||
| 
 |  | ||||||
|         <div className="d-flex mb-2 align-items-start"> |  | ||||||
|           <div style={{ minWidth: "100px" }}> |  | ||||||
|             <p className="m-0 text-start">Description :</p> |  | ||||||
|         </div> |         </div> | ||||||
|  | 
 | ||||||
|  |         <div className="d-flex mb-2 align-items-start" style={{ marginLeft: "3rem" }}> | ||||||
|  |           <div className="d-flex" style={{ minWidth: "130px" }}> | ||||||
|  |             <span className="d-flex align-items-start"> | ||||||
|  |               <i className="bx bx-book me-1"></i> | ||||||
|  |               <span>Description</span> | ||||||
|  |             </span> | ||||||
|  |             <span style={{ marginLeft: "10px" }}>:</span> | ||||||
|  |           </div> | ||||||
|  | 
 | ||||||
|           <div className="text-start"> |           <div className="text-start"> | ||||||
|             {displayText} |             {displayText} | ||||||
|             {isLong && ( |             {isLong && ( | ||||||
|  |               <> | ||||||
|  |                 <br /> | ||||||
|                 <span |                 <span | ||||||
|                   onClick={toggleReadMore} |                   onClick={toggleReadMore} | ||||||
|                   className="text-primary mx-1 cursor-pointer" |                   className="text-primary mx-1 cursor-pointer" | ||||||
|                 > |                 > | ||||||
|                   {expanded ? "Read less" : "Read more"} |                   {expanded ? "Read less" : "Read more"} | ||||||
|                 </span> |                 </span> | ||||||
|  |               </> | ||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|         <hr className="my-1" /> |         <hr className="my-1" /> | ||||||
|         <NotesDirectory |         <NotesDirectory | ||||||
|           refetchProfile={refetch} |           refetchProfile={refetch} | ||||||
|           isLoading={loading} |           isLoading={loading} | ||||||
|           contactProfile={profileContact} |           contactProfile={profileContactState} | ||||||
|           setProfileContact={setProfileContact} |           setProfileContact={setProfileContactState} | ||||||
|         /> |         /> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -14,7 +14,11 @@ import useMaster, { | |||||||
| } from "../../hooks/masterHook/useMaster"; | } from "../../hooks/masterHook/useMaster"; | ||||||
| import { useDispatch, useSelector } from "react-redux"; | import { useDispatch, useSelector } from "react-redux"; | ||||||
| import { changeMaster } from "../../slices/localVariablesSlice"; | import { changeMaster } from "../../slices/localVariablesSlice"; | ||||||
| import { useBuckets, useOrganization } from "../../hooks/useDirectory"; | import { | ||||||
|  |   useBuckets, | ||||||
|  |   useDesignation, | ||||||
|  |   useOrganization, | ||||||
|  | } from "../../hooks/useDirectory"; | ||||||
| import { useProjects } from "../../hooks/useProjects"; | import { useProjects } from "../../hooks/useProjects"; | ||||||
| import SelectMultiple from "../common/SelectMultiple"; | import SelectMultiple from "../common/SelectMultiple"; | ||||||
| import { ContactSchema } from "./DirectorySchema"; | import { ContactSchema } from "./DirectorySchema"; | ||||||
| @ -35,7 +39,10 @@ const UpdateContact = ({ submitContact, existingContact, onCLosed }) => { | |||||||
|   const [IsSubmitting, setSubmitting] = useState(false); |   const [IsSubmitting, setSubmitting] = useState(false); | ||||||
|   const [isInitialized, setIsInitialized] = useState(false); |   const [isInitialized, setIsInitialized] = useState(false); | ||||||
|   const dispatch = useDispatch(); |   const dispatch = useDispatch(); | ||||||
|   const {organizationList} = useOrganization() |   const { organizationList } = useOrganization(); | ||||||
|  |   const { designationList } = useDesignation(); | ||||||
|  |   const [showSuggestions, setShowSuggestions] = useState(false); | ||||||
|  |   const [filteredDesignationList, setFilteredDesignationList] = useState([]); | ||||||
| 
 | 
 | ||||||
|   const methods = useForm({ |   const methods = useForm({ | ||||||
|     resolver: zodResolver(ContactSchema), |     resolver: zodResolver(ContactSchema), | ||||||
| @ -45,6 +52,7 @@ const UpdateContact = ({ submitContact, existingContact, onCLosed }) => { | |||||||
|       contactCategoryId: null, |       contactCategoryId: null, | ||||||
|       address: "", |       address: "", | ||||||
|       description: "", |       description: "", | ||||||
|  |       designation: "", | ||||||
|       projectIds: [], |       projectIds: [], | ||||||
|       contactEmails: [], |       contactEmails: [], | ||||||
|       contactPhones: [], |       contactPhones: [], | ||||||
| @ -95,6 +103,24 @@ const UpdateContact = ({ submitContact, existingContact, onCLosed }) => { | |||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   // handle logic when input of desgination is changed | ||||||
|  |   const handleDesignationChange = (e) => { | ||||||
|  |     const val = e.target.value; | ||||||
|  | 
 | ||||||
|  |     const matches = designationList.filter((org) => | ||||||
|  |       org.toLowerCase().includes(val.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |     setFilteredDesignationList(matches); | ||||||
|  |     setShowSuggestions(true); | ||||||
|  |     setTimeout(() => setShowSuggestions(false), 5000); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   // handle logic when designation is selected | ||||||
|  |   const handleSelectDesignation = (val) => { | ||||||
|  |     setShowSuggestions(false); | ||||||
|  |     setValue("designation", val); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   const watchBucketIds = watch("bucketIds"); |   const watchBucketIds = watch("bucketIds"); | ||||||
| 
 | 
 | ||||||
|   const toggleBucketId = (id) => { |   const toggleBucketId = (id) => { | ||||||
| @ -119,17 +145,13 @@ const cleaned = { | |||||||
|         .filter((e) => e.emailAddress?.trim() !== "") |         .filter((e) => e.emailAddress?.trim() !== "") | ||||||
|         .map((email, index) => { |         .map((email, index) => { | ||||||
|           const existingEmail = existingContact.contactEmails?.[index]; |           const existingEmail = existingContact.contactEmails?.[index]; | ||||||
|       return existingEmail |           return existingEmail ? { ...email, id: existingEmail.id } : email; | ||||||
|         ? { ...email, id: existingEmail.id }  |  | ||||||
|         : email; |  | ||||||
|         }), |         }), | ||||||
|       contactPhones: (data.contactPhones || []) |       contactPhones: (data.contactPhones || []) | ||||||
|         .filter((p) => p.phoneNumber?.trim() !== "") |         .filter((p) => p.phoneNumber?.trim() !== "") | ||||||
|         .map((phone, index) => { |         .map((phone, index) => { | ||||||
|           const existingPhone = existingContact.contactPhones?.[index]; |           const existingPhone = existingContact.contactPhones?.[index]; | ||||||
|       return existingPhone |           return existingPhone ? { ...phone, id: existingPhone.id } : phone; | ||||||
|         ? { ...phone, id: existingPhone.id } |  | ||||||
|         : phone; |  | ||||||
|         }), |         }), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| @ -137,9 +159,8 @@ setSubmitting(true); | |||||||
|     await submitContact({ ...cleaned, id: existingContact.id }); |     await submitContact({ ...cleaned, id: existingContact.id }); | ||||||
| 
 | 
 | ||||||
|     setSubmitting(false); |     setSubmitting(false); | ||||||
|      |  | ||||||
|   }; |   }; | ||||||
|  const orgValue = watch("organization") |   const orgValue = watch("organization"); | ||||||
|   const handleClosed = () => { |   const handleClosed = () => { | ||||||
|     onCLosed(); |     onCLosed(); | ||||||
|   }; |   }; | ||||||
| @ -158,25 +179,31 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|         contactCategoryId: existingContact.contactCategory?.id || null, |         contactCategoryId: existingContact.contactCategory?.id || null, | ||||||
|         address: existingContact.address || "", |         address: existingContact.address || "", | ||||||
|         description: existingContact.description || "", |         description: existingContact.description || "", | ||||||
|  |         designation: existingContact.designation || "", | ||||||
|         projectIds: existingContact.projectIds || null, |         projectIds: existingContact.projectIds || null, | ||||||
|         tags: existingContact.tags || [], |         tags: existingContact.tags || [], | ||||||
|         bucketIds: existingContact.bucketIds || [], |         bucketIds: existingContact.bucketIds || [], | ||||||
|       }); |       }); | ||||||
| 
 | 
 | ||||||
|       if (!existingContact.contactPhones || existingContact.contactPhones.length === 0) { |       if ( | ||||||
|  |         !existingContact.contactPhones || | ||||||
|  |         existingContact.contactPhones.length === 0 | ||||||
|  |       ) { | ||||||
|         appendPhone({ label: "Office", phoneNumber: "" }); |         appendPhone({ label: "Office", phoneNumber: "" }); | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|     if (!existingContact.contactEmails || existingContact.contactEmails.length === 0) { |       if ( | ||||||
|  |         !existingContact.contactEmails || | ||||||
|  |         existingContact.contactEmails.length === 0 | ||||||
|  |       ) { | ||||||
|         appendEmail({ label: "Work", emailAddress: "" }); |         appendEmail({ label: "Work", emailAddress: "" }); | ||||||
|       } |       } | ||||||
|       setIsInitialized(true) |       setIsInitialized(true); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // return()=> reset() |     // return()=> reset() | ||||||
|   }, [existingContact, buckets, projects]); |   }, [existingContact, buckets, projects]); | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|   return ( |   return ( | ||||||
|     <FormProvider {...methods}> |     <FormProvider {...methods}> | ||||||
|       <form className="p-2 p-sm-0" onSubmit={handleSubmit(onSubmit)}> |       <form className="p-2 p-sm-0" onSubmit={handleSubmit(onSubmit)}> | ||||||
| @ -195,7 +222,6 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           |  | ||||||
|           <div className="col-md-6  text-start"> |           <div className="col-md-6  text-start"> | ||||||
|             <label className="form-label">Organization</label> |             <label className="form-label">Organization</label> | ||||||
|             <InputSuggestions |             <InputSuggestions | ||||||
| @ -211,6 +237,55 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |         <div className="row mt-1"> | ||||||
|  |           <div className="col-md-6  text-start"> | ||||||
|  |             <label className="form-label">Designation</label> | ||||||
|  |             <input | ||||||
|  |               className="form-control form-control-sm" | ||||||
|  |               {...register("designation")} | ||||||
|  |               onChange={handleDesignationChange} | ||||||
|  |             /> | ||||||
|  |             {showSuggestions && filteredDesignationList.length > 0 && ( | ||||||
|  |               <ul | ||||||
|  |                 className="list-group shadow-sm position-absolute w-50 bg-white border  zindex-tooltip" | ||||||
|  |                 style={{ | ||||||
|  |                   maxHeight: "180px", | ||||||
|  |                   overflowY: "auto", | ||||||
|  |                   marginTop: "2px", | ||||||
|  |                   zIndex: 1000, | ||||||
|  |                   borderRadius: "0px", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 {filteredDesignationList.map((designation) => ( | ||||||
|  |                   <li | ||||||
|  |                     key={designation} | ||||||
|  |                     className="list-group-item list-group-item-action border-none " | ||||||
|  |                     style={{ | ||||||
|  |                       cursor: "pointer", | ||||||
|  |                       padding: "5px 12px", | ||||||
|  |                       fontSize: "14px", | ||||||
|  |                       transition: "background-color 0.2s", | ||||||
|  |                     }} | ||||||
|  |                     onMouseDown={() => handleSelectDesignation(designation)} | ||||||
|  |                     onMouseEnter={(e) => | ||||||
|  |                       (e.currentTarget.style.backgroundColor = "#f8f9fa") | ||||||
|  |                     } | ||||||
|  |                     onMouseLeave={(e) => | ||||||
|  |                       (e.currentTarget.style.backgroundColor = "transparent") | ||||||
|  |                     } | ||||||
|  |                   > | ||||||
|  |                     {designation} | ||||||
|  |                   </li> | ||||||
|  |                 ))} | ||||||
|  |               </ul> | ||||||
|  |             )} | ||||||
|  |             {errors.designation && ( | ||||||
|  |               <small className="danger-text"> | ||||||
|  |                 {errors.designation.message} | ||||||
|  |               </small> | ||||||
|  |             )} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|         <div className="row mt-1"> |         <div className="row mt-1"> | ||||||
|           <div className="col-md-6"> |           <div className="col-md-6"> | ||||||
|             {emailFields.map((field, index) => ( |             {emailFields.map((field, index) => ( | ||||||
| @ -250,8 +325,10 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
| 
 | 
 | ||||||
|                       //   style={{ width: "24px", height: "24px" }} |                       //   style={{ width: "24px", height: "24px" }} | ||||||
|                       // > |                       // > | ||||||
|                         <i className="bx bx-plus-circle bx-xs ms-1 cursor-pointer text-primary" onClick={handleAddEmail}/> |                       <i | ||||||
|                   |                         className="bx bx-plus-circle bx-xs ms-1 cursor-pointer text-primary" | ||||||
|  |                         onClick={handleAddEmail} | ||||||
|  |                       /> | ||||||
|                     ) : ( |                     ) : ( | ||||||
|                       // <button |                       // <button | ||||||
|                       //   type="button" |                       //   type="button" | ||||||
| @ -259,8 +336,10 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|                       //   onClick={() => removeEmail(index)} |                       //   onClick={() => removeEmail(index)} | ||||||
|                       //   style={{ width: "24px", height: "24px" }} |                       //   style={{ width: "24px", height: "24px" }} | ||||||
|                       // > |                       // > | ||||||
|                         <i className="bx bx-minus-circle bx-xs ms-1 cursor-pointer text-danger" onClick={() => removeEmail(index)}/> |                       <i | ||||||
|                |                         className="bx bx-minus-circle bx-xs ms-1 cursor-pointer text-danger" | ||||||
|  |                         onClick={() => removeEmail(index)} | ||||||
|  |                       /> | ||||||
|                     )} |                     )} | ||||||
|                   </div> |                   </div> | ||||||
|                   {errors.contactEmails?.[index]?.emailAddress && ( |                   {errors.contactEmails?.[index]?.emailAddress && ( | ||||||
| @ -310,7 +389,10 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|                       //   onClick={handleAddPhone} |                       //   onClick={handleAddPhone} | ||||||
|                       //   style={{ width: "24px", height: "24px" }} |                       //   style={{ width: "24px", height: "24px" }} | ||||||
|                       // > |                       // > | ||||||
|                         <i className="bx bx-plus-circle bx-xs ms-1 cursor-pointer text-primary" onClick={handleAddPhone} /> |                       <i | ||||||
|  |                         className="bx bx-plus-circle bx-xs ms-1 cursor-pointer text-primary" | ||||||
|  |                         onClick={handleAddPhone} | ||||||
|  |                       /> | ||||||
|                     ) : ( |                     ) : ( | ||||||
|                       // <button |                       // <button | ||||||
|                       //   type="button" |                       //   type="button" | ||||||
| @ -318,7 +400,10 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|                       //   onClick={() => removePhone(index)} |                       //   onClick={() => removePhone(index)} | ||||||
|                       //   style={{ width: "24px", height: "24px" }} |                       //   style={{ width: "24px", height: "24px" }} | ||||||
|                       // > |                       // > | ||||||
|                         <i className="bx bx-minus-circle bx-xs ms-1 cursor-pointer text-danger" onClick={() => removePhone(index)} /> |                       <i | ||||||
|  |                         className="bx bx-minus-circle bx-xs ms-1 cursor-pointer text-danger" | ||||||
|  |                         onClick={() => removePhone(index)} | ||||||
|  |                       /> | ||||||
|                     )} |                     )} | ||||||
|                   </div> |                   </div> | ||||||
|                   {errors.contactPhones?.[index]?.phoneNumber && ( |                   {errors.contactPhones?.[index]?.phoneNumber && ( | ||||||
| @ -445,7 +530,11 @@ await submitContact({ ...cleaned, id: existingContact.id }); | |||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|         <div className="d-flex justify-content-center gap-1 py-2"> |         <div className="d-flex justify-content-center gap-1 py-2"> | ||||||
|           <button className="btn btn-sm btn-primary" type="submit" disabled={IsSubmitting}> |           <button | ||||||
|  |             className="btn btn-sm btn-primary" | ||||||
|  |             type="submit" | ||||||
|  |             disabled={IsSubmitting} | ||||||
|  |           > | ||||||
|             {IsSubmitting ? "Please Wait..." : "Update"} |             {IsSubmitting ? "Please Wait..." : "Update"} | ||||||
|           </button> |           </button> | ||||||
|           <button |           <button | ||||||
|  | |||||||
| @ -469,7 +469,7 @@ const { mutate: updateEmployee, isPending } = useUpdateEmployee(); | |||||||
|         </div> |         </div> | ||||||
|         <div className="row mb-3"> |         <div className="row mb-3"> | ||||||
|           <div className="col-sm-4"> |           <div className="col-sm-4"> | ||||||
|             <div className="form-text text-start">Role</div> |             <div className="form-text text-start">Official Designation</div> | ||||||
|             <div className="input-group input-group-merge "> |             <div className="input-group input-group-merge "> | ||||||
|               <select |               <select | ||||||
|                 className="form-select form-select-sm" |                 className="form-select form-select-sm" | ||||||
|  | |||||||
| @ -1,9 +1,9 @@ | |||||||
| 
 |  | ||||||
| import getGreetingMessage from "../../utils/greetingHandler"; | import getGreetingMessage from "../../utils/greetingHandler"; | ||||||
| import { | import { | ||||||
|   cacheData, |   cacheData, | ||||||
|   clearAllCache, |   clearAllCache, | ||||||
|   getCachedData, |   getCachedData, | ||||||
|  |   useSelectedproject, | ||||||
| } from "../../slices/apiDataManager"; | } from "../../slices/apiDataManager"; | ||||||
| import AuthRepository from "../../repositories/AuthRepository"; | import AuthRepository from "../../repositories/AuthRepository"; | ||||||
| import { useDispatch, useSelector } from "react-redux"; | import { useDispatch, useSelector } from "react-redux"; | ||||||
| @ -30,7 +30,15 @@ const Header = () => { | |||||||
| 
 | 
 | ||||||
|   const isDirectoryPath = /^\/directory$/.test(location.pathname); |   const isDirectoryPath = /^\/directory$/.test(location.pathname); | ||||||
|   const isProjectPath = /^\/projects$/.test(location.pathname); |   const isProjectPath = /^\/projects$/.test(location.pathname); | ||||||
|     const isDashboard = /^\/dashboard$/.test(location.pathname) || /^\/$/.test(location.pathname) ; |   const isDashboard = | ||||||
|  |     /^\/dashboard$/.test(location.pathname) || /^\/$/.test(location.pathname); | ||||||
|  | 
 | ||||||
|  |   const allowedProjectStatusIds = [ | ||||||
|  |     "603e994b-a27f-4e5d-a251-f3d69b0498ba", | ||||||
|  |     "cdad86aa-8a56-4ff4-b633-9c629057dfef", | ||||||
|  |     "b74da4c2-d07e-46f2-9919-e75e49b12731", | ||||||
|  |   ]; | ||||||
|  | 
 | ||||||
|   const getRole = (roles, joRoleId) => { |   const getRole = (roles, joRoleId) => { | ||||||
|     if (!Array.isArray(roles)) return "User"; |     if (!Array.isArray(roles)) return "User"; | ||||||
|     let role = roles.find((role) => role.id === joRoleId); |     let role = roles.find((role) => role.id === joRoleId); | ||||||
| @ -49,7 +57,7 @@ const Header = () => { | |||||||
|       }; |       }; | ||||||
| 
 | 
 | ||||||
|       AuthRepository.logout(data) |       AuthRepository.logout(data) | ||||||
|         .then((response) => { |         .then(() => { | ||||||
|           localStorage.removeItem("jwtToken"); |           localStorage.removeItem("jwtToken"); | ||||||
|           localStorage.removeItem("refreshToken"); |           localStorage.removeItem("refreshToken"); | ||||||
|           localStorage.removeItem("user"); |           localStorage.removeItem("user"); | ||||||
| @ -57,11 +65,11 @@ const Header = () => { | |||||||
|           clearAllCache(); |           clearAllCache(); | ||||||
|           window.location.href = "/auth/login"; |           window.location.href = "/auth/login"; | ||||||
|         }) |         }) | ||||||
|         .catch((error) => { |         .catch(() => { | ||||||
|           // Even if logout API fails, clear local storage and redirect |  | ||||||
|           localStorage.removeItem("jwtToken"); |           localStorage.removeItem("jwtToken"); | ||||||
|           localStorage.removeItem("refreshToken"); |           localStorage.removeItem("refreshToken"); | ||||||
|           localStorage.removeItem("user"); |           localStorage.removeItem("user"); | ||||||
|  |           localStorage.clear(); | ||||||
|           clearAllCache(); |           clearAllCache(); | ||||||
|           window.location.href = "/auth/login"; |           window.location.href = "/auth/login"; | ||||||
|         }); |         }); | ||||||
| @ -79,22 +87,30 @@ const Header = () => { | |||||||
| 
 | 
 | ||||||
|   const { projectNames, loading: projectLoading, fetchData } = useProjectName(); |   const { projectNames, loading: projectLoading, fetchData } = useProjectName(); | ||||||
| 
 | 
 | ||||||
|   const selectedProject = useSelector( |   const selectedProject = useSelectedproject(); | ||||||
|     (store) => store.localVariables.projectId | 
 | ||||||
|  |   const projectsForDropdown = isDashboard | ||||||
|  |     ? projectNames | ||||||
|  |     : projectNames?.filter(project => | ||||||
|  |         allowedProjectStatusIds.includes(project.projectStatusId) | ||||||
|       ); |       ); | ||||||
| 
 | 
 | ||||||
|   // Determine the display text for the project dropdown |   let currentProjectDisplayName; | ||||||
|   let displayText = "All Projects"; |   if (projectLoading) { | ||||||
|  |     currentProjectDisplayName = "Loading..."; | ||||||
|  |   } else if (!projectNames || projectNames.length === 0) { | ||||||
|  |     currentProjectDisplayName = "No Projects Assigned"; | ||||||
|  |   } else if (projectNames.length === 1) { | ||||||
|  |     currentProjectDisplayName = projectNames[0].name; | ||||||
|  |   } else { | ||||||
|     if (selectedProject === null) { |     if (selectedProject === null) { | ||||||
|     displayText = "All Projects"; |       currentProjectDisplayName = "All Projects"; | ||||||
|   } else if (selectedProject) { |     } else { | ||||||
|     const selectedProjectObj = projectNames?.find( |       const selectedProjectObj = projectNames.find( | ||||||
|         (p) => p?.id === selectedProject |         (p) => p?.id === selectedProject | ||||||
|       ); |       ); | ||||||
|     // Fallback to selectedProject ID if name not found during loading or mismatch |       currentProjectDisplayName = selectedProjectObj ? selectedProjectObj.name : "All Projects"; | ||||||
|     displayText = selectedProjectObj ? selectedProjectObj.name : selectedProject; |     } | ||||||
|   } else if (projectLoading) { |  | ||||||
|     displayText = "Loading..."; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const { openChangePassword } = useChangePassword(); |   const { openChangePassword } = useChangePassword(); | ||||||
| @ -106,17 +122,18 @@ const Header = () => { | |||||||
|       selectedProject === undefined && |       selectedProject === undefined && | ||||||
|       !getCachedData("hasReceived") |       !getCachedData("hasReceived") | ||||||
|     ) { |     ) { | ||||||
|  |       if (projectNames.length === 1) { | ||||||
|  |         dispatch(setProjectId(projectNames[0]?.id || null)); | ||||||
|  |       } else { | ||||||
|         if (isDashboard) { |         if (isDashboard) { | ||||||
|           dispatch(setProjectId(null)); |           dispatch(setProjectId(null)); | ||||||
|         } else { |         } else { | ||||||
|          dispatch(setProjectId(projectNames[0]?.id)); |           const firstAllowedProject = projectNames.find(project => allowedProjectStatusIds.includes(project.projectStatusId)); | ||||||
|  |           dispatch(setProjectId(firstAllowedProject?.id || null)); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|   }, [projectNames, selectedProject, dispatch]); |     } | ||||||
| 
 |   }, [projectNames, selectedProject, dispatch, isDashboard]); | ||||||
| 
 |  | ||||||
|   /** Check if current page is project details page or directory page */ |  | ||||||
|   // const isProjectPath = /^\/projects\/[a-f0-9-]{36}$/.test(location.pathname); |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|   const handler = useCallback( |   const handler = useCallback( | ||||||
| @ -161,13 +178,14 @@ const Header = () => { | |||||||
|   }, [handler, newProjectHandler]); |   }, [handler, newProjectHandler]); | ||||||
| 
 | 
 | ||||||
|   const handleProjectChange = (project) => { |   const handleProjectChange = (project) => { | ||||||
|     if(isProjectPath){ |     dispatch(setProjectId(project)); | ||||||
|      dispatch(setProjectId(project)) | 
 | ||||||
|      navigate("/projects/details") |     if (isProjectPath && project !== null) { | ||||||
|     } else{ |       navigate("/projects/details"); | ||||||
|       dispatch(setProjectId(project)) |  | ||||||
|     } |  | ||||||
|     } |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const shouldShowDropdown = projectNames && projectNames.length > 1; | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <nav |     <nav | ||||||
| @ -190,33 +208,37 @@ const Header = () => { | |||||||
|           <div className="align-items-center"> |           <div className="align-items-center"> | ||||||
|             <i className="rounded-circle bx bx-building-house bx-sm-lg bx-md me-2"></i> |             <i className="rounded-circle bx bx-building-house bx-sm-lg bx-md me-2"></i> | ||||||
|             <div className="btn-group"> |             <div className="btn-group"> | ||||||
|  |               {shouldShowDropdown ? ( | ||||||
|                 <button |                 <button | ||||||
|                 className={`btn btn-sm-sm btn-xl ${projectNames.length > 0 ? "dropdown-toggle" : "" |                   className={`btn btn-sm-sm btn-xl dropdown-toggle px-1`} | ||||||
|                   } px-1`} |  | ||||||
|                   type="button" |                   type="button" | ||||||
|                   data-bs-toggle="dropdown" |                   data-bs-toggle="dropdown" | ||||||
|                   aria-expanded="false" |                   aria-expanded="false" | ||||||
|                 > |                 > | ||||||
|                 {displayText} |                   {currentProjectDisplayName} | ||||||
|                 </button> |                 </button> | ||||||
|  |               ) : ( | ||||||
|  |                 <span className="btn btn-sm-sm btn-xl px-1"> | ||||||
|  |                   {currentProjectDisplayName} | ||||||
|  |                 </span> | ||||||
|  |               )} | ||||||
| 
 | 
 | ||||||
|               {projectNames.length > 0 && ( |               {shouldShowDropdown && projectsForDropdown && projectsForDropdown.length > 0 && ( | ||||||
|                 <ul |                 <ul | ||||||
|                   className="dropdown-menu" |                   className="dropdown-menu" | ||||||
|                   style={{ overflow: "auto", maxHeight: "300px" }} |                   style={{ overflow: "auto", maxHeight: "300px" }} | ||||||
|                 > |                 > | ||||||
|              |  | ||||||
|                   {isDashboard && ( |                   {isDashboard && ( | ||||||
|                     <li> |                     <li> | ||||||
|                       <button |                       <button | ||||||
|                         className="dropdown-item" |                         className="dropdown-item" | ||||||
|                         onClick={() => dispatch(setProjectId(null))} |                         onClick={() => handleProjectChange(null)} | ||||||
|                       > |                       > | ||||||
|                         All Projects |                         All Projects | ||||||
|                       </button> |                       </button> | ||||||
|                     </li> |                     </li> | ||||||
|                   )} |                   )} | ||||||
|                   {[...projectNames] |                   {[...projectsForDropdown] | ||||||
|                     .sort((a, b) => a?.name?.localeCompare(b.name)) |                     .sort((a, b) => a?.name?.localeCompare(b.name)) | ||||||
|                     .map((project) => ( |                     .map((project) => ( | ||||||
|                       <li key={project?.id}> |                       <li key={project?.id}> | ||||||
| @ -239,7 +261,6 @@ const Header = () => { | |||||||
|           </div> |           </div> | ||||||
|         )} |         )} | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|         <ul className="navbar-nav flex-row align-items-center ms-md-auto"> |         <ul className="navbar-nav flex-row align-items-center ms-md-auto"> | ||||||
|           <li className="nav-item dropdown-shortcuts navbar-dropdown dropdown me-2 me-xl-0"> |           <li className="nav-item dropdown-shortcuts navbar-dropdown dropdown me-2 me-xl-0"> | ||||||
|             <a |             <a | ||||||
|  | |||||||
| @ -8,6 +8,7 @@ import {MANAGE_PROJECT} from "../../utils/constants"; | |||||||
| import GlobalModel from "../common/GlobalModel"; | import GlobalModel from "../common/GlobalModel"; | ||||||
| import ManageProjectInfo from "./ManageProjectInfo"; | import ManageProjectInfo from "./ManageProjectInfo"; | ||||||
| import { useQueryClient } from "@tanstack/react-query"; | import { useQueryClient } from "@tanstack/react-query"; | ||||||
|  | import { useSelectedproject } from "../../slices/apiDataManager"; | ||||||
| 
 | 
 | ||||||
| const AboutProject = () => { | const AboutProject = () => { | ||||||
|   const [IsOpenModal, setIsOpenModal] = useState(false); |   const [IsOpenModal, setIsOpenModal] = useState(false); | ||||||
| @ -19,7 +20,8 @@ const AboutProject = () => { | |||||||
|   const ClientQuery = useQueryClient(); |   const ClientQuery = useQueryClient(); | ||||||
| 
 | 
 | ||||||
|   // *** MODIFIED LINE: Get projectId from Redux store using useSelector *** |   // *** MODIFIED LINE: Get projectId from Redux store using useSelector *** | ||||||
|   const projectId = useSelector((store) => store.localVariables.projectId); |   // const projectId = useSelector((store) => store.localVariables.projectId); | ||||||
|  |   const projectId = useSelectedproject(); | ||||||
| 
 | 
 | ||||||
|   const manageProject = useHasUserPermission(MANAGE_PROJECT); |   const manageProject = useHasUserPermission(MANAGE_PROJECT); | ||||||
|   const { projects_Details, isLoading, error, refetch } = useProjectDetails(projectId); // Pass projectId from useSelector |   const { projects_Details, isLoading, error, refetch } = useProjectDetails(projectId); // Pass projectId from useSelector | ||||||
| @ -51,71 +53,84 @@ const AboutProject = () => { | |||||||
|         <> |         <> | ||||||
|           <div className="card mb-6"> |           <div className="card mb-6"> | ||||||
|             <div className="card-header text-start"> |             <div className="card-header text-start"> | ||||||
|             <h6 className="card-action-title mb-0"> |               <h6 className="card-action-title mb-0 ps-1"> | ||||||
|                 {" "} |                 {" "} | ||||||
|                 <i className="fa fa-building rounded-circle text-primary"></i> |                 <i className="fa fa-building rounded-circle text-primary"></i> | ||||||
|                 <span className="ms-2">Project Profile</span> |                 <span className="ms-2">Project Profile</span> | ||||||
|               </h6> |               </h6> | ||||||
|             </div> |             </div> | ||||||
|             <div className="card-body"> |             <div className="card-body"> | ||||||
|             <ul className="list-unstyled my-3 ps-2"> |               <ul className="list-unstyled my-3 ps-0"> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|  |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> {/* Adjust width as needed for alignment */} | ||||||
|                     <i className="bx bx-cog"></i> |                     <i className="bx bx-cog"></i> | ||||||
|                 <span className="fw-medium mx-2">Name:</span>{" "} |                     <span className="fw-medium mx-2">Name:</span> | ||||||
|  |                   </div> | ||||||
|                   <span>{projects_Details.name}</span> |                   <span>{projects_Details.name}</span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|  |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> | ||||||
|                     <i className="bx bx-fingerprint"></i> |                     <i className="bx bx-fingerprint"></i> | ||||||
|                 <span className="fw-medium mx-2">Nick Name:</span>{" "} |                     <span className="fw-medium mx-2">Nick Name:</span> | ||||||
|  |                   </div> | ||||||
|                   <span>{projects_Details.shortName}</span> |                   <span>{projects_Details.shortName}</span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|  |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> | ||||||
|                     <i className="bx bx-check"></i> |                     <i className="bx bx-check"></i> | ||||||
|                 <span className="fw-medium mx-2">Start Date:</span>{" "} |                     <span className="fw-medium mx-2">Start Date:</span> | ||||||
|  |                   </div> | ||||||
|                   <span> |                   <span> | ||||||
|                     {projects_Details.startDate |                     {projects_Details.startDate | ||||||
|                       ? moment(projects_Details.startDate).format("DD-MMM-YYYY") |                       ? moment(projects_Details.startDate).format("DD-MMM-YYYY") | ||||||
|                       : "N/A"} |                       : "N/A"} | ||||||
|                   </span> |                   </span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|                 <i className="bx bx-stop-circle"></i>{" "} |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> | ||||||
|                 <span className="fw-medium mx-2">End Date:</span>{" "} |                     <i className="bx bx-stop-circle"></i> | ||||||
|  |                     <span className="fw-medium mx-2">End Date:</span> | ||||||
|  |                   </div> | ||||||
|                   <span> |                   <span> | ||||||
|                     {projects_Details.endDate |                     {projects_Details.endDate | ||||||
|                       ? moment(projects_Details.endDate).format("DD-MMM-YYYY") |                       ? moment(projects_Details.endDate).format("DD-MMM-YYYY") | ||||||
|                       : "N/A"} |                       : "N/A"} | ||||||
|                   </span> |                   </span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|  |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> | ||||||
|                     <i className="bx bx-trophy"></i> |                     <i className="bx bx-trophy"></i> | ||||||
|                 <span className="fw-medium mx-2">Status:</span>{" "} |                     <span className="fw-medium mx-2">Status:</span> | ||||||
|                 <span>{projects_Details?.projectStatus?.status}</span> |                   </div> | ||||||
|  |                   <span>{projects_Details?.projectStatus?.status.replace(/\s/g, '')}</span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex align-items-center mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|  |                   <div className="d-flex align-items-center" style={{ width: '120px' }}> | ||||||
|                     <i className="bx bx-user"></i> |                     <i className="bx bx-user"></i> | ||||||
|                 <span className="fw-medium mx-2">Contact:</span>{" "} |                     <span className="fw-medium mx-2">Contact:</span> | ||||||
|  |                   </div> | ||||||
|                   <span>{projects_Details.contactPerson}</span> |                   <span>{projects_Details.contactPerson}</span> | ||||||
|                 </li> |                 </li> | ||||||
|               <li className="d-flex flex-column align-items-start mb-3"> |                 <li className="d-flex mb-3"> | ||||||
|                 <div className="d-flex align-items-center"> |                   {/* Label section with icon */} | ||||||
|                   <i className="bx bx-flag"></i> |                   <div className="d-flex align-items-start" style={{ minWidth: "120px" }}> | ||||||
|                   <span className="fw-medium mx-2">Address:</span> |                     <i className="bx bx-flag mt-1"></i> | ||||||
|                   {projects_Details.projectAddress?.length <= 20 && ( |                     <span className="fw-medium mx-2 text-nowrap">Address:</span> | ||||||
|                     <span>{projects_Details.projectAddress}</span> |                   </div> | ||||||
|                   )} | 
 | ||||||
|  |                   {/* Content section that wraps nicely */} | ||||||
|  |                   <div className="flex-grow-1 text-start text-wrap"> | ||||||
|  |                     {projects_Details.projectAddress} | ||||||
|                   </div> |                   </div> | ||||||
|                 {projects_Details.projectAddress?.length > 20 && ( |  | ||||||
|                   <div className="ms-4 text-start">{projects_Details.projectAddress}</div> |  | ||||||
|                 )} |  | ||||||
|                 </li> |                 </li> | ||||||
| 
 | 
 | ||||||
|               <li className="d-flex justify-content-center mb-3"> | 
 | ||||||
|  | 
 | ||||||
|  |                 <li className="d-flex justify-content-center mt-4"> {/* Added mt-4 for some top margin */} | ||||||
|                   {manageProject && ( |                   {manageProject && ( | ||||||
|                     <button |                     <button | ||||||
|                       type="button" |                       type="button" | ||||||
|                     className={`btn btn-sm btn-primary ${ |                       className={`btn btn-sm btn-primary ${!manageProject && "d-none" | ||||||
|                       !manageProject && "d-none" |  | ||||||
|                         }`} |                         }`} | ||||||
|                       data-bs-toggle="modal" |                       data-bs-toggle="modal" | ||||||
|                       data-bs-target="#edit-project-modal" |                       data-bs-target="#edit-project-modal" | ||||||
| @ -126,8 +141,10 @@ const AboutProject = () => { | |||||||
|                   )} |                   )} | ||||||
|                 </li> |                 </li> | ||||||
|               </ul> |               </ul> | ||||||
|  | 
 | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|  | 
 | ||||||
|         </> |         </> | ||||||
|       )} |       )} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -48,6 +48,9 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|   const infoRef = useRef(null); |   const infoRef = useRef(null); | ||||||
|   const infoRef1 = useRef(null); |   const infoRef1 = useRef(null); | ||||||
| 
 | 
 | ||||||
|  |   // State for search term | ||||||
|  |   const [searchTerm, setSearchTerm] = useState(""); | ||||||
|  | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (typeof bootstrap !== "undefined") { |     if (typeof bootstrap !== "undefined") { | ||||||
|       if (infoRef.current) { |       if (infoRef.current) { | ||||||
| @ -83,7 +86,10 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|   const { loading } = useMaster(); |   const { loading } = useMaster(); | ||||||
|   const { data: jobRoleData } = useMaster(); |   const { data: jobRoleData } = useMaster(); | ||||||
| 
 | 
 | ||||||
|   const [selectedRole, setSelectedRole] = useState("all"); |   // Changed to an array to hold multiple selected roles | ||||||
|  |   const [selectedRoles, setSelectedRoles] = useState(["all"]); | ||||||
|  |   // Changed to an array to hold multiple selected roles | ||||||
|  |   // const [selectedRoles, setSelectedRoles] = useState(["all"]); | ||||||
|   const [displayedSelection, setDisplayedSelection] = useState(""); |   const [displayedSelection, setDisplayedSelection] = useState(""); | ||||||
|   const { |   const { | ||||||
|     handleSubmit, |     handleSubmit, | ||||||
| @ -121,21 +127,80 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     dispatch(changeMaster("Job Role")); |     dispatch(changeMaster("Job Role")); | ||||||
| 
 |     // Initial state should reflect "All Roles" selected | ||||||
|     return () => setSelectedRole("all"); |     setSelectedRoles(["all"]); | ||||||
|   }, [dispatch]); |   }, [dispatch]); | ||||||
| 
 | 
 | ||||||
|   const handleRoleChange = (event) => { |   // Modified handleRoleChange to handle multiple selections | ||||||
|     setSelectedRole(event.target.value); |   const handleRoleChange = (event, roleId) => { | ||||||
|  |     // If 'all' is selected, clear other selections | ||||||
|  |     if (roleId === "all") { | ||||||
|  |       setSelectedRoles(["all"]); | ||||||
|  |     } else { | ||||||
|  |       setSelectedRoles((prevSelectedRoles) => { | ||||||
|  |         // If "all" was previously selected, remove it | ||||||
|  |         const newRoles = prevSelectedRoles.filter((role) => role !== "all"); | ||||||
|  |         if (newRoles.includes(roleId)) { | ||||||
|  |           // If role is already selected, unselect it | ||||||
|  |           return newRoles.filter((id) => id !== roleId); | ||||||
|  |         } else { | ||||||
|  |           // If role is not selected, add it | ||||||
|  |           return [...newRoles, roleId]; | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const filteredEmployees = |   useEffect(() => { | ||||||
|     selectedRole === "all" |     // Update displayedSelection based on selectedRoles | ||||||
|       ? employees |     if (selectedRoles.includes("all")) { | ||||||
|       : employees?.filter( |       setDisplayedSelection("All Roles"); | ||||||
|           (emp) => String(emp.jobRoleId || "") === selectedRole |     } else if (selectedRoles.length > 0) { | ||||||
|  |       const selectedRoleNames = selectedRoles.map(roleId => { | ||||||
|  |         const role = jobRoleData?.find(r => String(r.id) === roleId); | ||||||
|  |         return role ? role.name : ''; | ||||||
|  |       }).filter(Boolean); // Filter out empty strings for roles not found | ||||||
|  |       setDisplayedSelection(selectedRoleNames.join(', ')); | ||||||
|  |     } else { | ||||||
|  |       setDisplayedSelection("Select Roles"); | ||||||
|  |     } | ||||||
|  |   }, [selectedRoles, jobRoleData]); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   const handleSearchChange = (event) => { | ||||||
|  |     setSearchTerm(event.target.value); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   // Filter employees first by role, then by search term AND job role name | ||||||
|  |   const filteredEmployees = employees?.filter((emp) => { | ||||||
|  |     const matchesRole = | ||||||
|  |       selectedRoles.includes("all") || selectedRoles.includes(String(emp.jobRoleId)); | ||||||
|  |     // Convert both first and last names and job role name to lowercase for case-insensitive matching | ||||||
|  |     const fullName = `${emp.firstName} ${emp.lastName}`.toLowerCase(); | ||||||
|  | 
 | ||||||
|  |     const jobRoleName = jobRoleData?.find((role) => role.id === emp.jobRoleId)?.name?.toLowerCase() || ""; | ||||||
|  | 
 | ||||||
|  |     const searchLower = searchTerm.toLowerCase(); | ||||||
|  |     // Check if the full name OR job role name includes the search term | ||||||
|  |     const matchesSearch = fullName.includes(searchLower) || jobRoleName.includes(searchLower); | ||||||
|  |     return matchesRole && matchesSearch; | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   // Determine unique job role IDs from the filtered employees (for dropdown options) | ||||||
|  |   const uniqueJobRoleIdsInFilteredEmployees = new Set( | ||||||
|  |     employees?.map(emp => emp.jobRoleId).filter(Boolean) | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|  |   // Filter jobRoleData to only include roles present in the uniqueJobRoleIdsInFilteredEmployees | ||||||
|  |   const jobRolesForDropdown = jobRoleData?.filter(role => | ||||||
|  |     uniqueJobRoleIdsInFilteredEmployees.has(role.id) | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   // Calculate the count of selected roles for display | ||||||
|  |   const selectedRolesCount = selectedRoles.includes("all") | ||||||
|  |     ? 0 // "All Roles" doesn't contribute to a specific count | ||||||
|  |     : selectedRoles.length; | ||||||
|  | 
 | ||||||
|   const onSubmit = (data) => { |   const onSubmit = (data) => { | ||||||
|     const selectedEmployeeIds = data.selectedEmployees; |     const selectedEmployeeIds = data.selectedEmployees; | ||||||
| 
 | 
 | ||||||
| @ -159,6 +224,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|     reset(); |     reset(); | ||||||
|     onClose(); |     onClose(); | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|    <div className="fs-5 text-dark text-center d-flex align-items-center justify-content-center flex-wrap"> |    <div className="fs-5 text-dark text-center d-flex align-items-center justify-content-center flex-wrap"> | ||||||
|   <p className="align-items-center flex-wrap m-0 ">Assign Task</p> |   <p className="align-items-center flex-wrap m-0 ">Assign Task</p> | ||||||
| @ -166,7 +232,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|     <div className="mb-1"> |     <div className="mb-1"> | ||||||
|       <p className="mb-0"> |       <p className="mb-0"> | ||||||
|         <span className="text-dark text-start d-flex align-items-center flex-wrap form-text"> |         <span className="text-dark text-start d-flex align-items-center flex-wrap form-text"> | ||||||
|               <span className="me-2 m-0 font-bold">Work Location :</span> |           <span className="me-2 m-0 fw-bold">Work Location :</span> {/* Changed font-bold to fw-bold */} | ||||||
|           {[ |           {[ | ||||||
|             assignData?.building?.buildingName, |             assignData?.building?.buildingName, | ||||||
|             assignData?.floor?.floorName, |             assignData?.floor?.floorName, | ||||||
| @ -192,50 +258,92 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|               <div className="form-text text-start"> |               <div className="form-text text-start"> | ||||||
|                 <div className="d-flex align-items-center form-text fs-7"> |                 <div className="d-flex align-items-center form-text fs-7"> | ||||||
|                   <span className="text-dark">Select Team</span> |                   <span className="text-dark">Select Team</span> | ||||||
|                       <div className="me-2">{displayedSelection}</div> |                   <div className="dropdown position-relative d-inline-block"> | ||||||
|                     <a |                     <a | ||||||
|                         className="dropdown-toggle hide-arrow cursor-pointer" |                       className={`dropdown-toggle hide-arrow cursor-pointer ${ | ||||||
|  |                         selectedRoles.includes("all") || selectedRoles.length === 0 | ||||||
|  |                           ? "text-secondary" | ||||||
|  |                           : "text-primary" | ||||||
|  |                       }`} | ||||||
|                       data-bs-toggle="dropdown" |                       data-bs-toggle="dropdown" | ||||||
|  |                       role="button" | ||||||
|                       aria-expanded="false" |                       aria-expanded="false" | ||||||
|                     > |                     > | ||||||
|                         <i className="bx bx-filter bx-lg text-primary"></i> |                       <i className="bx bx-slider-alt ms-2"></i> | ||||||
|                     </a> |                     </a> | ||||||
| 
 | 
 | ||||||
|                       <ul className="dropdown-menu p-2 text-capitalize"> |                     {/* Badge */} | ||||||
|                         <li key="all"> |                     {selectedRolesCount > 0 && ( | ||||||
|                           <button |                       <span | ||||||
|                             type="button" |                         className="position-absolute top-0 start-100 translate-middle badge rounded-circle bg-warning text-white" | ||||||
|                             className="dropdown-item py-1" |                         style={{ | ||||||
|                             onClick={() => |                           fontSize: "0.65rem", | ||||||
|                               handleRoleChange({ |                           minWidth: "18px", | ||||||
|                                 target: { value: "all" }, |                           height: "18px", | ||||||
|                               }) |                           padding: "0", | ||||||
|                             } |                           lineHeight: "18px", | ||||||
|  |                           textAlign: "center", | ||||||
|  |                           zIndex: 10, | ||||||
|  |                         }} | ||||||
|                       > |                       > | ||||||
|  |                         {selectedRolesCount} | ||||||
|  |                       </span> | ||||||
|  |                     )} | ||||||
|  | 
 | ||||||
|  |                     {/* Dropdown Menu - Corrected: Removed duplicate ul block */} | ||||||
|  |                     <ul className="dropdown-menu p-2 text-capitalize" style={{ maxHeight: "300px", overflowY: "auto" }}> | ||||||
|  |                       <li> {/* Changed key="all" to a unique key if possible, or keep it if "all" is a unique identifier */} | ||||||
|  |                         <div className="form-check dropdown-item py-0"> | ||||||
|  |                           <input | ||||||
|  |                             className="form-check-input" | ||||||
|  |                             type="checkbox" | ||||||
|  |                             id="checkboxAllRoles" // Unique ID | ||||||
|  |                             value="all" | ||||||
|  |                             checked={selectedRoles.includes("all")} | ||||||
|  |                             onChange={(e) => handleRoleChange(e, e.target.value)} | ||||||
|  |                           /> | ||||||
|  |                           <label className="form-check-label ms-2" htmlFor="checkboxAllRoles"> | ||||||
|                             All Roles |                             All Roles | ||||||
|                           </button> |                           </label> | ||||||
|  |                         </div> | ||||||
|                       </li> |                       </li> | ||||||
|                         {jobRoleData?.map((user) => ( | 
 | ||||||
|                           <li key={user.id}> |                       {jobRolesForDropdown?.map((role) => ( | ||||||
|                             <button |                         <li key={role.id}> | ||||||
|                               type="button" |                           <div className="form-check dropdown-item py-0"> | ||||||
|                               className="dropdown-item py-1" |                             <input | ||||||
|                               value={user.id} |                               className="form-check-input" | ||||||
|                               onClick={handleRoleChange} |                               type="checkbox" | ||||||
|                             > |                               id={`checkboxRole-${role.id}`} // Unique ID | ||||||
|                               {user.name} |                               value={role.id} | ||||||
|                             </button> |                               checked={selectedRoles.includes(String(role.id))} | ||||||
|  |                               onChange={(e) => handleRoleChange(e, e.target.value)} | ||||||
|  |                             /> | ||||||
|  |                             <label className="form-check-label ms-2" htmlFor={`checkboxRole-${role.id}`}> | ||||||
|  |                               {role.name} | ||||||
|  |                             </label> | ||||||
|  |                           </div> | ||||||
|                         </li> |                         </li> | ||||||
|                       ))} |                       ))} | ||||||
|                     </ul> |                     </ul> | ||||||
|                   </div> |                   </div> | ||||||
|  |                   <input | ||||||
|  |                     type="text" | ||||||
|  |                     className="form-control form-control-sm ms-auto mb-2 mt-2" | ||||||
|  |                     placeholder="Search employees or roles..." | ||||||
|  |                     value={searchTerm} | ||||||
|  |                     onChange={handleSearchChange} | ||||||
|  |                     style={{ maxWidth: "200px" }} | ||||||
|  |                   /> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
| 
 |           </div> | ||||||
|               <div className="row"> |           <div | ||||||
|                 <div className="col-12 h-sm-25 overflow-auto mt-2"> |             className="col-12 mt-2" | ||||||
|                   {selectedRole !== "" && ( |             style={{ maxHeight: "280px", overflowY: "auto", overflowX: "hidden" }} | ||||||
|  |           > | ||||||
|  |             {selectedRoles?.length > 0 && ( | ||||||
|               <div className="row"> |               <div className="row"> | ||||||
|                 {employeeLoading ? ( |                 {employeeLoading ? ( | ||||||
|                   <div className="col-12"> |                   <div className="col-12"> | ||||||
| @ -261,7 +369,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                                 {...field} |                                 {...field} | ||||||
|                                 className="form-check-input me-1 mt-1" |                                 className="form-check-input me-1 mt-1" | ||||||
|                                 type="checkbox" |                                 type="checkbox" | ||||||
|                                       id={`employee-${emp?.id}`} |                                 id={`employee-${emp?.id}`} // Unique ID | ||||||
|                                 value={emp.id} |                                 value={emp.id} | ||||||
|                                 checked={field.value?.includes(emp.id)} |                                 checked={field.value?.includes(emp.id)} | ||||||
|                                 onChange={(e) => { |                                 onChange={(e) => { | ||||||
| @ -297,14 +405,13 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                 ) : ( |                 ) : ( | ||||||
|                   <div className="col-12"> |                   <div className="col-12"> | ||||||
|                     <p className="text-center"> |                     <p className="text-center"> | ||||||
|                             No employees found for the selected role. |                       No employees found for the selected role(s). | ||||||
|                     </p> |                     </p> | ||||||
|                   </div> |                   </div> | ||||||
|                 )} |                 )} | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|           </div> |           </div> | ||||||
|               </div> |  | ||||||
| 
 | 
 | ||||||
|           <div |           <div | ||||||
|             className="col-12 h-25 overflow-auto" |             className="col-12 h-25 overflow-auto" | ||||||
| @ -322,10 +429,11 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                           className="badge rounded-pill bg-label-primary d-inline-flex align-items-center me-1 mb-1" |                           className="badge rounded-pill bg-label-primary d-inline-flex align-items-center me-1 mb-1" | ||||||
|                         > |                         > | ||||||
|                           {emp.firstName} {emp.lastName} |                           {emp.firstName} {emp.lastName} | ||||||
|                               <p |                           {/* Changed p tag to button for semantic correctness and accessibility */} | ||||||
|  |                           <button | ||||||
|                             type="button" |                             type="button" | ||||||
|                                 className=" btn-close-white p-0 m-0" |                             className="btn-close btn-close-white ms-1" // Added ms-1 for spacing, removed p-0 m-0 | ||||||
|                                 aria-label="Close" |                             aria-label="Remove employee" // More descriptive aria-label | ||||||
|                             onClick={() => { |                             onClick={() => { | ||||||
|                               const updatedSelected = watch( |                               const updatedSelected = watch( | ||||||
|                                 "selectedEmployees" |                                 "selectedEmployees" | ||||||
| @ -338,7 +446,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                             }} |                             }} | ||||||
|                           > |                           > | ||||||
|                             <i className="icon-base bx bx-x icon-md"></i> |                             <i className="icon-base bx bx-x icon-md"></i> | ||||||
|                               </p> |                           </button> | ||||||
|                         </span> |                         </span> | ||||||
|                       ) |                       ) | ||||||
|                     ); |                     ); | ||||||
| @ -358,12 +466,12 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|             <div className="form-check form-check-inline mt-3 px-1"> |             <div className="form-check form-check-inline mt-3 px-1"> | ||||||
|               <label |               <label | ||||||
|                 className="form-text text-dark align-items-center d-flex" |                 className="form-text text-dark align-items-center d-flex" | ||||||
|                     htmlFor="inlineCheckbox1" |                 htmlFor="inlineCheckbox1" // This htmlFor isn't linked to a checkbox in this context | ||||||
|               > |               > | ||||||
|                 Pending Task of Activity : |                 Pending Task of Activity : | ||||||
|                 <label |                 <label | ||||||
|                   className="form-check-label fs-7 ms-4" |                   className="form-check-label fs-7 ms-4" | ||||||
|                       htmlFor="inlineCheckbox1" |                   htmlFor="inlineCheckbox1" // This htmlFor isn't linked to a checkbox in this context | ||||||
|                 > |                 > | ||||||
|                   <strong> |                   <strong> | ||||||
|                     {assignData?.workItem?.plannedWork - |                     {assignData?.workItem?.plannedWork - | ||||||
| @ -410,7 +518,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|             <div className="form-check form-check-inline mt-2 px-1 mb-2 text-start"> |             <div className="form-check form-check-inline mt-2 px-1 mb-2 text-start"> | ||||||
|               <label |               <label | ||||||
|                 className="text-dark text-start d-flex align-items-center flex-wrap form-text" |                 className="text-dark text-start d-flex align-items-center flex-wrap form-text" | ||||||
|                     htmlFor="inlineCheckbox1" |                 htmlFor="targetForTodayInput" // Added a unique htmlFor for clarity | ||||||
|               > |               > | ||||||
|                 <span>Target for Today</span>  |                 <span>Target for Today</span>  | ||||||
|                 <span style={{ marginLeft: "46px" }}>:</span> |                 <span style={{ marginLeft: "46px" }}>:</span> | ||||||
| @ -429,14 +537,17 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                       type="text" |                       type="text" | ||||||
|                       className="form-control form-control-sm" |                       className="form-control form-control-sm" | ||||||
|                       {...field} |                       {...field} | ||||||
|                           id="defaultFormControlInput" |                       id="defaultFormControlInput" // Consider a more descriptive ID if used elsewhere | ||||||
|                       aria-describedby="defaultFormControlHelp" |                       aria-describedby="defaultFormControlHelp" | ||||||
|                     /> |                     /> | ||||||
|                         <span style={{ paddingLeft: "6px" }}> |                     <span style={{ paddingLeft: "6px", whiteSpace: "nowrap" }}> | ||||||
|  |                       <u> | ||||||
|  |                         {" "} | ||||||
|                         { |                         { | ||||||
|                             assignData?.workItem?.workItem?.activityMaster |                           assignData?.workItem?.activityMaster | ||||||
|                             ?.unitOfMeasurement |                             ?.unitOfMeasurement | ||||||
|                         } |                         } | ||||||
|  |                       </u> | ||||||
|                     </span> |                     </span> | ||||||
|                     <div |                     <div | ||||||
|                       style={{ |                       style={{ | ||||||
| @ -484,7 +595,6 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|                 className="position-absolute bg-white border p-2 rounded shadow" |                 className="position-absolute bg-white border p-2 rounded shadow" | ||||||
|                 style={{ zIndex: 10, marginLeft: "10px" }} |                 style={{ zIndex: 10, marginLeft: "10px" }} | ||||||
|               > |               > | ||||||
|                     {/* Add your help content here */} |  | ||||||
|                 <p className="mb-0"> |                 <p className="mb-0"> | ||||||
|                   Enter the target value for today's task. |                   Enter the target value for today's task. | ||||||
|                 </p> |                 </p> | ||||||
| @ -493,8 +603,8 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <label |           <label | ||||||
|                 className="form-text fs-7 m-1 text-lg text-dark" |             className="form-text fs-7 m-1 text-dark" // Removed duplicate htmlFor and text-lg | ||||||
|                 htmlFor="descriptionTextarea" // Changed htmlFor for better accessibility |             htmlFor="descriptionTextarea" | ||||||
|           > |           > | ||||||
|             Description |             Description | ||||||
|           </label> |           </label> | ||||||
| @ -505,7 +615,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|               <textarea |               <textarea | ||||||
|                 {...field} |                 {...field} | ||||||
|                 className="form-control" |                 className="form-control" | ||||||
|                     id="descriptionTextarea" // Changed id for better accessibility |                 id="descriptionTextarea" // Unique ID | ||||||
|                 rows="2" |                 rows="2" | ||||||
|               /> |               /> | ||||||
|             )} |             )} | ||||||
| @ -515,7 +625,6 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|           )} |           )} | ||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|             {/* Submit and Cancel buttons */} |  | ||||||
|         <div className="col-12 d-flex justify-content-center align-items-center gap-sm-6 gap-8 text-center mt-1"> |         <div className="col-12 d-flex justify-content-center align-items-center gap-sm-6 gap-8 text-center mt-1"> | ||||||
|           <button |           <button | ||||||
|             type="submit" |             type="submit" | ||||||
| @ -538,8 +647,7 @@ const AssignTask = ({ assignData, onClose, setAssigned }) => { | |||||||
|       </form> |       </form> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|     </div> | </div>  ); | ||||||
|   ); |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export default AssignTask; | export default AssignTask; | ||||||
| @ -15,6 +15,7 @@ import { | |||||||
|   cacheData, |   cacheData, | ||||||
|   clearCacheKey, |   clearCacheKey, | ||||||
|   getCachedData, |   getCachedData, | ||||||
|  |   useSelectedproject, | ||||||
| } from "../../slices/apiDataManager"; | } from "../../slices/apiDataManager"; | ||||||
| import { useProjectDetails, useProjectInfra } from "../../hooks/useProjects"; | import { useProjectDetails, useProjectInfra } from "../../hooks/useProjects"; | ||||||
| import { useDispatch, useSelector } from "react-redux"; | import { useDispatch, useSelector } from "react-redux"; | ||||||
| @ -25,7 +26,8 @@ import GlobalModel from "../common/GlobalModel"; | |||||||
| 
 | 
 | ||||||
| const ProjectInfra = ( {data, onDataChange, eachSiteEngineer} ) => | const ProjectInfra = ( {data, onDataChange, eachSiteEngineer} ) => | ||||||
| { | { | ||||||
|   const projectId = useSelector((store)=>store.localVariables.projectId) |   // const projectId = useSelector((store)=>store.localVariables.projectId) | ||||||
|  |   const projectId = useSelectedproject(); | ||||||
|   const reloadedData = useSelector((store) => store.localVariables.reload); |   const reloadedData = useSelector((store) => store.localVariables.reload); | ||||||
|   const [ expandedBuildings, setExpandedBuildings ] = useState( [] ); |   const [ expandedBuildings, setExpandedBuildings ] = useState( [] ); | ||||||
|    const {projectInfra,isLoading,error} = useProjectInfra(projectId) |    const {projectInfra,isLoading,error} = useProjectInfra(projectId) | ||||||
|  | |||||||
| @ -49,20 +49,7 @@ const ProjectNav = ({ onPillClick, activePill }) => { | |||||||
|           </a> |           </a> | ||||||
|         </li> |         </li> | ||||||
|        |        | ||||||
|         <li className="nav-item"> |          | ||||||
|           <a |  | ||||||
|             className={`nav-link ${ |  | ||||||
|               activePill === "imagegallary" ? "active" : "" |  | ||||||
|             } fs-6`} |  | ||||||
|             href="#" |  | ||||||
|             onClick={(e) => { |  | ||||||
|               e.preventDefault(); // Prevent page reload |  | ||||||
|               onPillClick("imagegallary"); |  | ||||||
|             }} |  | ||||||
|           > |  | ||||||
|            <i className='bx bxs-cog bx-sm me-1_5'></i> <span className="d-none d-md-inline">project Setup</span> |  | ||||||
|           </a> |  | ||||||
|         </li> |  | ||||||
|         {(DirAdmin || DireManager || DirUser) && ( |         {(DirAdmin || DireManager || DirUser) && ( | ||||||
|             <li className="nav-item"> |             <li className="nav-item"> | ||||||
|           <a |           <a | ||||||
| @ -77,6 +64,20 @@ const ProjectNav = ({ onPillClick, activePill }) => { | |||||||
|           </a> |           </a> | ||||||
|         </li> |         </li> | ||||||
|         )} |         )} | ||||||
|  |         <li className="nav-item"> | ||||||
|  |           <a | ||||||
|  |             className={`nav-link ${ | ||||||
|  |               activePill === "imagegallary" ? "active" : "" | ||||||
|  |             } fs-6`} | ||||||
|  |             href="#" | ||||||
|  |             onClick={(e) => { | ||||||
|  |               e.preventDefault(); // Prevent page reload | ||||||
|  |               onPillClick("imagegallary"); | ||||||
|  |             }} | ||||||
|  |           > | ||||||
|  |            <i className='bx bxs-cog bx-sm me-1_5'></i> <span className="d-none d-md-inline">project Setup</span> | ||||||
|  |           </a> | ||||||
|  |         </li> | ||||||
|        |        | ||||||
|       </ul> |       </ul> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -15,11 +15,13 @@ import { ASSIGN_TO_PROJECT } from "../../utils/constants"; | |||||||
| import ConfirmModal from "../common/ConfirmModal"; | import ConfirmModal from "../common/ConfirmModal"; | ||||||
| import eventBus from "../../services/eventBus"; | import eventBus from "../../services/eventBus"; | ||||||
| import {useEmployeesByProjectAllocated, useManageProjectAllocation} from "../../hooks/useProjects"; | import {useEmployeesByProjectAllocated, useManageProjectAllocation} from "../../hooks/useProjects"; | ||||||
|  | import { useSelectedproject } from "../../slices/apiDataManager"; | ||||||
| 
 | 
 | ||||||
| const Teams = () => | const Teams = () => | ||||||
| { | { | ||||||
|   // const {projectId} = useParams() |   // const {projectId} = useParams() | ||||||
|   const projectId = useSelector((store)=>store.localVariables.projectId) |   // const projectId = useSelector((store)=>store.localVariables.projectId) | ||||||
|  |   const projectId = useSelectedproject(); | ||||||
|   const dispatch = useDispatch(); |   const dispatch = useDispatch(); | ||||||
| 
 | 
 | ||||||
|   const { data, loading } = useMaster(); |   const { data, loading } = useMaster(); | ||||||
| @ -293,7 +295,7 @@ const { | |||||||
|                             <div className="d-flex flex-column"> |                             <div className="d-flex flex-column"> | ||||||
|                               <a |                               <a | ||||||
|                                 onClick={() => |                                 onClick={() => | ||||||
|                                   navigate(`/employee/${item.employeeId}?for=account`) |                                   navigate(`/employee/${item.employeeId}?for=attendance`) | ||||||
|                                 } |                                 } | ||||||
|                                 className="text-heading text-truncate cursor-pointer" |                                 className="text-heading text-truncate cursor-pointer" | ||||||
|                               > |                               > | ||||||
|  | |||||||
| @ -141,54 +141,58 @@ const CreateRole = ({ modalType, onClose }) => { | |||||||
|         )} |         )} | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div className="col-12 col-md-12  border"> |       <div | ||||||
|  |         className="border rounded px-3" | ||||||
|  |         style={{ | ||||||
|  |           maxHeight: "350px", | ||||||
|  |           overflowY: "auto", | ||||||
|  |           overflowX: "hidden", // Prevent bottom scrollbar | ||||||
|  |         }} | ||||||
|  |       > | ||||||
|         {masterFeatures.map((feature, featureIndex) => ( |         {masterFeatures.map((feature, featureIndex) => ( | ||||||
|           <React.Fragment key={feature.id}> |           <React.Fragment key={feature.id}> | ||||||
|             <div |             <div className="row my-1"> | ||||||
|               className="row my-1" |               {/* Feature Title */} | ||||||
|               key={feature.id} |               <div className="col-12 text-start fw-semibold mb-2"> | ||||||
|               style={{ marginLeft: "0px" }} |                 {feature.name} | ||||||
|             > |  | ||||||
|               <div |  | ||||||
|                 className="col-12 col-md-3 d-flex text-start align-items-start" |  | ||||||
|                 style={{ wordWrap: "break-word" }} |  | ||||||
|               > |  | ||||||
|                 <span className="fs">{feature.name}</span> |  | ||||||
|               </div> |               </div> | ||||||
| 
 | 
 | ||||||
|               <div className="col-12 col-md-1"></div> |               {/* Permissions Grid */} | ||||||
| 
 |               <div className="col-12"> | ||||||
|               <div className="col-12 col-md-8 d-flex justify-content-start align-items-center flex-wrap"> |                 <div className="row"> | ||||||
|                   {feature.featurePermissions.map((perm, permIndex) => { |                   {feature.featurePermissions.map((perm, permIndex) => { | ||||||
|                   const refIndex = featureIndex * 10 + permIndex; |                     const refIndex = featureIndex * 100 + permIndex; | ||||||
|                     return ( |                     return ( | ||||||
|                     <div className="d-flex me-3 mb-2" key={perm.id}> |                       <div | ||||||
|                       <label className="form-check-label" htmlFor={perm.id}> |                         className="col-12 col-sm-6 col-md-4 mb-3" | ||||||
|  |                         key={perm.id} | ||||||
|  |                       > | ||||||
|  |                         <div className="d-flex align-items-start"> | ||||||
|  |                           <label | ||||||
|  |                             className="form-check-label d-flex align-items-center" | ||||||
|  |                             htmlFor={perm.id} | ||||||
|  |                           > | ||||||
|                             <input |                             <input | ||||||
|                               type="checkbox" |                               type="checkbox" | ||||||
|                           className="form-check-input mx-2" |                               className="form-check-input me-2" | ||||||
|                               id={perm.id} |                               id={perm.id} | ||||||
|                               value={perm.id} |                               value={perm.id} | ||||||
|                               {...register("selectedPermissions")} |                               {...register("selectedPermissions")} | ||||||
|                             /> |                             /> | ||||||
|                             {perm.name} |                             {perm.name} | ||||||
|                           </label> |                           </label> | ||||||
|                       <div style={{ display: "flex", alignItems: "center" }}> | 
 | ||||||
|  |                           {/* Info icon */} | ||||||
|  |                           <div className="ms-1 d-flex align-items-center"> | ||||||
|                             <div |                             <div | ||||||
|                           key={refIndex} |  | ||||||
|                               ref={(el) => (popoverRefs.current[refIndex] = el)} |                               ref={(el) => (popoverRefs.current[refIndex] = el)} | ||||||
|                               tabIndex="0" |                               tabIndex="0" | ||||||
|                           className="d-flex align-items-center avatar-group justify-content-center" |                               className="d-flex align-items-center justify-content-center" | ||||||
|                               data-bs-toggle="popover" |                               data-bs-toggle="popover" | ||||||
|                           refIndex |  | ||||||
|                               data-bs-trigger="focus" |                               data-bs-trigger="focus" | ||||||
|                               data-bs-placement="right" |                               data-bs-placement="right" | ||||||
|                               data-bs-html="true" |                               data-bs-html="true" | ||||||
|                           data-bs-content={` |                               data-bs-content={`<div class="border border-secondary rounded custom-popover p-2 px-3">${perm.description}</div>`} | ||||||
|                                       <div class="border border-secondary rounded custom-popover p-2 px-3"> |  | ||||||
|                                         ${perm.description} |  | ||||||
|                                       </div> |  | ||||||
|                                     `} |  | ||||||
|                             > |                             > | ||||||
|                                 |                                 | ||||||
|                               <svg |                               <svg | ||||||
| @ -205,19 +209,24 @@ const CreateRole = ({ modalType, onClose }) => { | |||||||
|                             </div> |                             </div> | ||||||
|                           </div> |                           </div> | ||||||
|                         </div> |                         </div> | ||||||
|  |                       </div> | ||||||
|                     ); |                     ); | ||||||
|                   })} |                   })} | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|             <hr className="hr my-1 py-1" /> |             </div> | ||||||
|  | 
 | ||||||
|  |             <hr className="my-2" /> | ||||||
|           </React.Fragment> |           </React.Fragment> | ||||||
|         ))} |         ))} | ||||||
|  | 
 | ||||||
|         {errors.selectedPermissions && ( |         {errors.selectedPermissions && ( | ||||||
|           <p className="text-danger">{errors.selectedPermissions.message}</p> |           <p className="text-danger">{errors.selectedPermissions.message}</p> | ||||||
|         )} |         )} | ||||||
|         {!masterFeatures && <p>Loading...</p>} |         {!masterFeatures && <p>Loading...</p>} | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|       {masterFeatures && ( |       {masterFeatures && ( | ||||||
|         <div className="col-12 text-center"> |         <div className="col-12 text-center"> | ||||||
|           <button type="submit" className="btn btn-sm btn-primary me-3"> |           <button type="submit" className="btn btn-sm btn-primary me-3"> | ||||||
|  | |||||||
| @ -9,13 +9,6 @@ import { cacheData, getCachedData } from "../../slices/apiDataManager"; | |||||||
| import showToast from "../../services/toastService"; | import showToast from "../../services/toastService"; | ||||||
| import { useUpdateApplicationRole } from "../../hooks/masterHook/useMaster"; | import { useUpdateApplicationRole } from "../../hooks/masterHook/useMaster"; | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| const updateSchema = z.object({ | const updateSchema = z.object({ | ||||||
|   role: z.string().min(1, { message: "Role is required" }), |   role: z.string().min(1, { message: "Role is required" }), | ||||||
|   description: z.string().min(1, { message: "Description is required" }) |   description: z.string().min(1, { message: "Description is required" }) | ||||||
| @ -25,8 +18,6 @@ const updateSchema = z.object({ | |||||||
|   }), |   }), | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| const EditMaster = ({ master, onClose }) => { | const EditMaster = ({ master, onClose }) => { | ||||||
|   const maxDescriptionLength = 255; |   const maxDescriptionLength = 255; | ||||||
|   const popoverRefs = useRef([]); |   const popoverRefs = useRef([]); | ||||||
| @ -199,77 +190,91 @@ useEffect(() => { | |||||||
|         )} |         )} | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div className="col-12 col-md-12 mx-2s" > |      <div className="col-12 text-start"> | ||||||
| 
 |   {/* Scrollable Container with Border */} | ||||||
|  |   <div | ||||||
|  |     className="border rounded p-3" | ||||||
|  |     style={{ | ||||||
|  |       maxHeight: "350px", | ||||||
|  |       overflowY: "auto", | ||||||
|  |       overflowX: "hidden", // Prevent horizontal scrollbar | ||||||
|  |       paddingRight: "10px", | ||||||
|  |     }} | ||||||
|  |   > | ||||||
|     {masterFeatures.map((feature, featureIndex) => ( |     {masterFeatures.map((feature, featureIndex) => ( | ||||||
|           <div className="row my-1" key={feature.id} style={{ marginLeft: "0px" }}> |       <div key={feature.id} className="mb-3"> | ||||||
|  |         {/* Feature Group Title */} | ||||||
|  |         <div className="fw-semibold mb-2">{feature.name}</div> | ||||||
| 
 | 
 | ||||||
|             <div className="col-12 col-md-3 d-flex text-start align-items-center" style={{ wordWrap: 'break-word' }}> |         {/* Permissions Grid */} | ||||||
|               <span className="fs">{feature.name}</span> |         <div className="row"> | ||||||
|             </div> |  | ||||||
|             <div className="col-12 col-md-1"> |  | ||||||
| 
 |  | ||||||
|             </div> |  | ||||||
|             <div className="col-12 col-md-8 d-flex justify-content-start align-items-center flex-wrap "> |  | ||||||
|           {feature.featurePermissions.map((perm, permIndex) => { |           {feature.featurePermissions.map((perm, permIndex) => { | ||||||
|                 const refIndex = (featureIndex * 10) + permIndex; |             const refIndex = featureIndex * 10 + permIndex; | ||||||
|             return ( |             return ( | ||||||
| 
 |               <div | ||||||
|                   <div className="d-flex me-3 mb-2" key={perm.id}> |                 key={perm.id} | ||||||
| 
 |                 className="col-12 col-sm-6 col-md-4 mb-3 d-flex align-items-start" | ||||||
|                     <label className="form-check-label" htmlFor={perm.id}> |               > | ||||||
|  |                 <label | ||||||
|  |                   className="form-check-label d-flex align-items-center" | ||||||
|  |                   htmlFor={perm.id} | ||||||
|  |                 > | ||||||
|                   <input |                   <input | ||||||
|                     type="checkbox" |                     type="checkbox" | ||||||
|                         className="form-check-input mx-2" |                     className="form-check-input me-2" | ||||||
|                     id={perm.id} |                     id={perm.id} | ||||||
|                     {...register(`permissions.${perm.id}`, { |                     {...register(`permissions.${perm.id}`, { | ||||||
|                           value: initialPermissions[perm.id] || false |                       value: initialPermissions[perm.id] || false, | ||||||
|                     })} |                     })} | ||||||
|                   /> |                   /> | ||||||
| 
 |  | ||||||
|                   {perm.name} |                   {perm.name} | ||||||
|                 </label> |                 </label> | ||||||
| 
 | 
 | ||||||
| 
 |                 {/* Info Icon */} | ||||||
|                     <div style={{ display: 'flex', alignItems: 'center' }}> |                 <div style={{ display: "flex", alignItems: "center" }}> | ||||||
|                   <div |                   <div | ||||||
|                         key={refIndex} |                     ref={(el) => (popoverRefs.current[refIndex] = el)} | ||||||
|                         ref={(el) => |  | ||||||
|                           (popoverRefs.current[refIndex] = el) |  | ||||||
|                         } |  | ||||||
|                     tabIndex="0" |                     tabIndex="0" | ||||||
|                         className="d-flex align-items-center avatar-group justify-content-center" |                     className="d-flex align-items-center justify-content-center" | ||||||
|                         data-bs-toggle="popover" refIndex |                     data-bs-toggle="popover" | ||||||
|                     data-bs-trigger="focus" |                     data-bs-trigger="focus" | ||||||
|                     data-bs-placement="right" |                     data-bs-placement="right" | ||||||
|                     data-bs-html="true" |                     data-bs-html="true" | ||||||
|                         data-bs-content={` |                     data-bs-content={`<div class="border border-secondary rounded custom-popover p-2 px-3">${perm.description}</div>`} | ||||||
|                                       <div class="border border-secondary rounded custom-popover p-2 px-3"> |  | ||||||
|                                         ${perm.description} |  | ||||||
|                                       </div> |  | ||||||
|                                     `} |  | ||||||
|                   > |                   > | ||||||
|                       |                       | ||||||
|                         <svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" className="bi bi-info-circle" viewBox="0 0 16 16"> |                     <svg | ||||||
|  |                       xmlns="http://www.w3.org/2000/svg" | ||||||
|  |                       width="13" | ||||||
|  |                       height="13" | ||||||
|  |                       fill="currentColor" | ||||||
|  |                       className="bi bi-info-circle" | ||||||
|  |                       viewBox="0 0 16 16" | ||||||
|  |                     > | ||||||
|                       <path d="M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z" /> |                       <path d="M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z" /> | ||||||
|                       <path d="m8.93 6.588-2.29.287-.082.38.45.083c.294.07.352.176.288.469l-.738 3.468c-.194.897.105 1.319.547 1.11l1.91-2.011c.241-.256.384-.592.287-.984-.172-.439-.58-.827-1.13-.967a.664.664 0 0 1-.58-.309l-.15-.241-.002-.002zM8 4c-.535 0-.943.372-.943.836 0 .464.408.836.943.836.535 0 .943-.372.943-.836 0-.464-.408-.836-.943-.836z" /> |                       <path d="m8.93 6.588-2.29.287-.082.38.45.083c.294.07.352.176.288.469l-.738 3.468c-.194.897.105 1.319.547 1.11l1.91-2.011c.241-.256.384-.592.287-.984-.172-.439-.58-.827-1.13-.967a.664.664 0 0 1-.58-.309l-.15-.241-.002-.002zM8 4c-.535 0-.943.372-.943.836 0 .464.408.836.943.836.535 0 .943-.372.943-.836 0-.464-.408-.836-.943-.836z" /> | ||||||
|                     </svg> |                     </svg> | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|                 ) |             ); | ||||||
|           })} |           })} | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|         </div> |         </div> | ||||||
|             <hr className="hr my-1 py-1" /> | 
 | ||||||
|  |         <hr className="my-2" /> | ||||||
|       </div> |       </div> | ||||||
|     ))} |     ))} | ||||||
|  |   </div> | ||||||
|  | 
 | ||||||
|  |   {/* Error Display */} | ||||||
|   {errors.permissions && ( |   {errors.permissions && ( | ||||||
|     <p className="text-danger">{errors.permissions.message}</p> |     <p className="text-danger">{errors.permissions.message}</p> | ||||||
|   )} |   )} | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|       <div className="col-12 text-center"> |       <div className="col-12 text-center"> | ||||||
|         <button type="submit" className="btn btn-sm btn-primary me-3"> {isLoading ? "Please Wait..." : "Submit"}</button> |         <button type="submit" className="btn btn-sm btn-primary me-3"> {isLoading ? "Please Wait..." : "Submit"}</button> | ||||||
|         <button |         <button | ||||||
|  | |||||||
| @ -104,17 +104,15 @@ export const useDashboardTeamsCardData = (projectId) => { | |||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const fetchTeamsData = async () => { |     const fetchTeamsData = async () => { | ||||||
|       if (!projectId) return; // ✅ Skip if projectId is not provided |  | ||||||
| 
 |  | ||||||
|       setLoading(true); |       setLoading(true); | ||||||
|       setError(""); |       setError(""); | ||||||
| 
 | 
 | ||||||
|       try { |       try { | ||||||
|         const response = await GlobalRepository.getDashboardTeamsCardData(projectId); |         const response = await GlobalRepository.getDashboardTeamsCardData(projectId); | ||||||
|         setTeamsData(response.data); // ✅ Handle undefined/null |         setTeamsData(response.data || {}); | ||||||
|       } catch (err) { |       } catch (err) { | ||||||
|         setError("Failed to fetch teams card data."); |         setError("Failed to fetch teams card data."); | ||||||
|         console.error(err); |         console.error("Error fetching teams card data:", err); | ||||||
|         setTeamsData({}); |         setTeamsData({}); | ||||||
|       } finally { |       } finally { | ||||||
|         setLoading(false); |         setLoading(false); | ||||||
|  | |||||||
| @ -177,3 +177,36 @@ export const useOrganization = () => { | |||||||
| 
 | 
 | ||||||
|   return { organizationList, loading, error }; |   return { organizationList, loading, error }; | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | export const useDesignation = () => { | ||||||
|  |   const [designationList, setDesignationList] = useState([]); | ||||||
|  |   const [loading, setLoading] = useState(false); | ||||||
|  |   const [error, setError] = useState(""); | ||||||
|  | 
 | ||||||
|  |   const fetchOrg = async () => { | ||||||
|  |     const cacheOrg = getCachedData("designation"); | ||||||
|  |     if (cacheOrg?.length != 0) { | ||||||
|  |       setLoading(true); | ||||||
|  |       try { | ||||||
|  |         const resp = await DirectoryRepository.GetDesignations(); | ||||||
|  |         cacheData("designation", resp.data); | ||||||
|  |         setDesignationList(resp.data); | ||||||
|  |         setLoading(false); | ||||||
|  |       } catch (error) { | ||||||
|  |         const msg = | ||||||
|  |           error?.response?.data?.message || | ||||||
|  |           error?.message || | ||||||
|  |           "Something went wrong"; | ||||||
|  |         setError(msg); | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       setDesignationList(cacheOrg); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     fetchOrg(); | ||||||
|  |   }, []); | ||||||
|  | 
 | ||||||
|  |   return { designationList, loading, error }; | ||||||
|  | }; | ||||||
|  | |||||||
| @ -8,33 +8,35 @@ import { | |||||||
| import Breadcrumb from "../../components/common/Breadcrumb"; | import Breadcrumb from "../../components/common/Breadcrumb"; | ||||||
| import AttendanceLog from "../../components/Activities/AttendcesLogs"; | import AttendanceLog from "../../components/Activities/AttendcesLogs"; | ||||||
| import Attendance from "../../components/Activities/Attendance"; | import Attendance from "../../components/Activities/Attendance"; | ||||||
|  | import AttendanceModel from "../../components/Activities/AttendanceModel"; | ||||||
| import showToast from "../../services/toastService"; | import showToast from "../../services/toastService"; | ||||||
| import Regularization from "../../components/Activities/Regularization"; | import Regularization from "../../components/Activities/Regularization"; | ||||||
| import { useAttendance } from "../../hooks/useAttendance"; | import { useAttendance } from "../../hooks/useAttendance"; | ||||||
| import { useDispatch, useSelector } from "react-redux"; | import { useDispatch, useSelector } from "react-redux"; | ||||||
| import { setProjectId } from "../../slices/localVariablesSlice"; | import { setProjectId } from "../../slices/localVariablesSlice"; | ||||||
| import { hasUserPermission } from "../../utils/authUtils"; | import { markCurrentAttendance } from "../../slices/apiSlice/attendanceAllSlice"; | ||||||
| import { useHasUserPermission } from "../../hooks/useHasUserPermission"; | import { useHasUserPermission } from "../../hooks/useHasUserPermission"; | ||||||
| import { REGULARIZE_ATTENDANCE } from "../../utils/constants"; | import { REGULARIZE_ATTENDANCE } from "../../utils/constants"; | ||||||
| import eventBus from "../../services/eventBus"; | import eventBus from "../../services/eventBus"; | ||||||
|  | import AttendanceRepository from "../../repositories/AttendanceRepository"; | ||||||
| import { useProjectName } from "../../hooks/useProjects"; | import { useProjectName } from "../../hooks/useProjects"; | ||||||
| import GlobalModel from "../../components/common/GlobalModel"; |  | ||||||
| import CheckCheckOutmodel from "../../components/Activities/CheckCheckOutForm"; |  | ||||||
| import AttendLogs from "../../components/Activities/AttendLogs"; |  | ||||||
| import { useQueryClient } from "@tanstack/react-query"; |  | ||||||
| 
 | 
 | ||||||
| const AttendancePage = () => { | const AttendancePage = () => { | ||||||
|   const [activeTab, setActiveTab] = useState("all"); |   const [activeTab, setActiveTab] = useState("all"); | ||||||
|   const [ShowPending, setShowPending] = useState(false); |   const [showPending, setShowPending] = useState(false); | ||||||
|   const queryClient = useQueryClient(); |   const [searchQuery, setSearchQuery] = useState(""); | ||||||
|   const loginUser = getCachedProfileData(); |   const loginUser = getCachedProfileData(); | ||||||
|   var selectedProject = useSelector((store) => store.localVariables.projectId); |   const selectedProject = useSelector((store) => store.localVariables.projectId); | ||||||
|   const dispatch = useDispatch(); |   const dispatch = useDispatch(); | ||||||
| 
 |   const { | ||||||
|  |     attendance, | ||||||
|  |     loading: attLoading, | ||||||
|  |     recall: attrecall, | ||||||
|  |   } = useAttendance(selectedProject); | ||||||
|   const [attendances, setAttendances] = useState(); |   const [attendances, setAttendances] = useState(); | ||||||
|   const [empRoles, setEmpRoles] = useState(null); |   const [empRoles, setEmpRoles] = useState(null); | ||||||
|   const [isCreateModalOpen, setIsCreateModalOpen] = useState(false); |   const [isCreateModalOpen, setIsCreateModalOpen] = useState(false); | ||||||
|   const [modelConfig, setModelConfig] = useState(); |   const [modelConfig, setModelConfig] = useState(null); // Initialize as null | ||||||
|   const DoRegularized = useHasUserPermission(REGULARIZE_ATTENDANCE); |   const DoRegularized = useHasUserPermission(REGULARIZE_ATTENDANCE); | ||||||
|   const { projectNames, loading: projectLoading, fetchData } = useProjectName(); |   const { projectNames, loading: projectLoading, fetchData } = useProjectName(); | ||||||
| 
 | 
 | ||||||
| @ -44,68 +46,172 @@ const AttendancePage = () => { | |||||||
|     date: new Date().toLocaleDateString(), |     date: new Date().toLocaleDateString(), | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   const handler = useCallback( | ||||||
|     if (selectedProject == null) { |     (msg) => { | ||||||
|       dispatch(setProjectId(projectNames[0]?.id)); |       if (selectedProject === msg.projectId) { | ||||||
|  |         const updatedAttendance = attendances | ||||||
|  |           ? attendances.map((item) => | ||||||
|  |               item.employeeId === msg.response.employeeId | ||||||
|  |                 ? { ...item, ...msg.response } | ||||||
|  |                 : item | ||||||
|  |             ) | ||||||
|  |           : [msg.response]; | ||||||
|  | 
 | ||||||
|  |         cacheData("Attendance", { | ||||||
|  |           data: updatedAttendance, | ||||||
|  |           projectId: selectedProject, | ||||||
|  |         }); | ||||||
|  |         setAttendances(updatedAttendance); | ||||||
|       } |       } | ||||||
|   }, []); |     }, | ||||||
|  |     [selectedProject, attendances] | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   const employeeHandler = useCallback( | ||||||
|  |     (msg) => { | ||||||
|  |       // This logic seems fine for refetching if an employee ID exists in current attendances | ||||||
|  |       if (attendances?.some((item) => item.employeeId === msg.employeeId)) { | ||||||
|  |         AttendanceRepository.getAttendance(selectedProject) | ||||||
|  |           .then((response) => { | ||||||
|  |             cacheData("Attendance", { data: response.data, selectedProject }); | ||||||
|  |             setAttendances(response.data); | ||||||
|  |           }) | ||||||
|  |           .catch((error) => { | ||||||
|  |             console.error(error); | ||||||
|  |           }); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     [selectedProject, attendances] | ||||||
|  |   ); | ||||||
| 
 | 
 | ||||||
|   const getRole = (roleId) => { |   const getRole = (roleId) => { | ||||||
|     if (!empRoles) return "Unassigned"; |     if (!empRoles) return "Unassigned"; | ||||||
|     if (!roleId) return "Unassigned"; |     if (!roleId) return "Unassigned"; | ||||||
|     const role = empRoles.find((b) => b.id == roleId); |     const role = empRoles.find((b) => b.id === roleId); | ||||||
|     return role ? role.role : "Unassigned"; |     return role ? role.role : "Unassigned"; | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const openModel = () => { |   // Simplified and moved modal opening logic | ||||||
|     setIsCreateModalOpen(true); |   const handleModalData = useCallback((employee) => { | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const handleModalData = (employee) => { |  | ||||||
|     setModelConfig(employee); |     setModelConfig(employee); | ||||||
|   }; |     setIsCreateModalOpen(true); // Open the modal directly when data is set | ||||||
|  |   }, []); | ||||||
| 
 | 
 | ||||||
|   const closeModal = () => { |   const closeModal = useCallback(() => { | ||||||
|     setModelConfig(null); |     setModelConfig(null); | ||||||
|     setIsCreateModalOpen(false); |     setIsCreateModalOpen(false); | ||||||
|   }; |     // Directly manipulating the DOM is generally not recommended in React. | ||||||
|  |     // React handles modal visibility via state. If you must, ensure it's | ||||||
|  |     // for external libraries or for very specific, controlled reasons. | ||||||
|  |     // For a typical Bootstrap modal, just setting `isCreateModalOpen` to false | ||||||
|  |     // should be enough if the modal component itself handles the Bootstrap classes. | ||||||
|  |     const modalElement = document.getElementById("check-Out-modal"); | ||||||
|  |     if (modalElement) { | ||||||
|  |       modalElement.classList.remove("show"); | ||||||
|  |       modalElement.style.display = "none"; | ||||||
|  |       document.body.classList.remove("modal-open"); | ||||||
|  |       const modalBackdrop = document.querySelector(".modal-backdrop"); | ||||||
|  |       if (modalBackdrop) { | ||||||
|  |         modalBackdrop.remove(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }, []); | ||||||
|  | 
 | ||||||
|  |   const handleSubmit = useCallback((formData) => { | ||||||
|  |     dispatch(markCurrentAttendance(formData)) | ||||||
|  |       .then((action) => { | ||||||
|  |         if (action.payload && action.payload.employeeId) { | ||||||
|  |           const updatedAttendance = attendances | ||||||
|  |             ? attendances.map((item) => | ||||||
|  |                 item.employeeId === action.payload.employeeId | ||||||
|  |                   ? { ...item, ...action.payload } | ||||||
|  |                   : item | ||||||
|  |               ) | ||||||
|  |             : [action.payload]; | ||||||
|  | 
 | ||||||
|  |           cacheData("Attendance", { | ||||||
|  |             data: updatedAttendance, | ||||||
|  |             projectId: selectedProject, | ||||||
|  |           }); | ||||||
|  |           setAttendances(updatedAttendance); | ||||||
|  |           showToast("Attendance Marked Successfully", "success"); | ||||||
|  |         } else { | ||||||
|  |           showToast("Failed to mark attendance: Invalid response", "error"); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .catch((error) => { | ||||||
|  |         showToast(error.message, "error"); | ||||||
|  |       }); | ||||||
|  |   }, [dispatch, attendances, selectedProject]); | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|   const handleToggle = (event) => { |   const handleToggle = (event) => { | ||||||
|     setShowOnlyCheckout(event.target.checked); |     setShowPending(event.target.checked); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (selectedProject === null && projectNames.length > 0) { | ||||||
|  |       dispatch(setProjectId(projectNames[0]?.id)); | ||||||
|  |     } | ||||||
|  |   }, [selectedProject, projectNames, dispatch]); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (modelConfig !== null) { |     setAttendances(attendance); | ||||||
|       openModel(); |   }, [attendance]); | ||||||
|  | 
 | ||||||
|  |   const filteredAndSearchedTodayAttendance = useCallback(() => { | ||||||
|  |     let currentData = attendances; | ||||||
|  | 
 | ||||||
|  |     if (showPending) { | ||||||
|  |       currentData = currentData?.filter( | ||||||
|  |         (att) => att?.checkInTime !== null && att?.checkOutTime === null | ||||||
|  |       ); | ||||||
|     } |     } | ||||||
|   }, [modelConfig, isCreateModalOpen]); | 
 | ||||||
|  |     if (searchQuery) { | ||||||
|  |       const lowerCaseSearchQuery = searchQuery.toLowerCase(); | ||||||
|  |       currentData = currentData?.filter((att) => { | ||||||
|  |         const fullName = [att.firstName, att.middleName, att.lastName] | ||||||
|  |           .filter(Boolean) | ||||||
|  |           .join(" ") | ||||||
|  |           .toLowerCase(); | ||||||
|  | 
 | ||||||
|  |         return ( | ||||||
|  |           att.employeeName?.toLowerCase().includes(lowerCaseSearchQuery) || | ||||||
|  |           att.employeeId?.toLowerCase().includes(lowerCaseSearchQuery) || | ||||||
|  |           fullName.includes(lowerCaseSearchQuery) | ||||||
|  |         ); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     return currentData; | ||||||
|  |   }, [attendances, showPending, searchQuery]); | ||||||
|  | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     eventBus.on("attendance", handler); | ||||||
|  |     return () => eventBus.off("attendance", handler); | ||||||
|  |   }, [handler]); | ||||||
|  | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     eventBus.on("employee", employeeHandler); | ||||||
|  |     return () => eventBus.off("employee", employeeHandler); | ||||||
|  |   }, [employeeHandler]); | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
| 
 |  | ||||||
|       {isCreateModalOpen && modelConfig && ( |       {isCreateModalOpen && modelConfig && ( | ||||||
|         <GlobalModel |         <div | ||||||
|           isOpen={isCreateModalOpen} |           className="modal fade show" | ||||||
|           size={modelConfig?.action === 6 && "lg"} |           style={{ display: "block" }} | ||||||
|           closeModal={closeModal} |           id="check-Out-modal" | ||||||
|  |           tabIndex="-1" | ||||||
|  |           aria-hidden="true" | ||||||
|         > |         > | ||||||
|           {(modelConfig?.action === 0 || |           <AttendanceModel | ||||||
|             modelConfig?.action === 1 || |             modelConfig={modelConfig} | ||||||
|             modelConfig?.action === 2) && ( |  | ||||||
|             <CheckCheckOutmodel |  | ||||||
|               modeldata={modelConfig} |  | ||||||
|             closeModal={closeModal} |             closeModal={closeModal} | ||||||
|  |             handleSubmitForm={handleSubmit} | ||||||
|           /> |           /> | ||||||
|           )} |         </div> | ||||||
|           {/* For view logs */} |  | ||||||
|           {modelConfig?.action === 6 && ( |  | ||||||
|             <AttendLogs Id={modelConfig?.id} closeModal={closeModal} /> |  | ||||||
|           )} |  | ||||||
|           {modelConfig?.action === 7 && ( |  | ||||||
|             <Confirmation closeModal={closeModal} /> |  | ||||||
|           )} |  | ||||||
|         </GlobalModel> |  | ||||||
|       )} |       )} | ||||||
| 
 | 
 | ||||||
|       <div className="container-fluid"> |       <div className="container-fluid"> | ||||||
| @ -116,13 +222,15 @@ const AttendancePage = () => { | |||||||
|           ]} |           ]} | ||||||
|         ></Breadcrumb> |         ></Breadcrumb> | ||||||
|         <div className="nav-align-top nav-tabs-shadow"> |         <div className="nav-align-top nav-tabs-shadow"> | ||||||
|           <ul className="nav nav-tabs" role="tablist"> |           <ul | ||||||
|  |             className="nav nav-tabs d-flex justify-content-between align-items-center" | ||||||
|  |             role="tablist" | ||||||
|  |           > | ||||||
|  |             <div className="d-flex"> | ||||||
|               <li className="nav-item"> |               <li className="nav-item"> | ||||||
|                 <button |                 <button | ||||||
|                   type="button" |                   type="button" | ||||||
|                 className={`nav-link ${ |                   className={`nav-link ${activeTab === "all" ? "active" : ""} fs-6`} | ||||||
|                   activeTab === "all" ? "active" : "" |  | ||||||
|                 } fs-6`} |  | ||||||
|                   onClick={() => setActiveTab("all")} |                   onClick={() => setActiveTab("all")} | ||||||
|                   data-bs-toggle="tab" |                   data-bs-toggle="tab" | ||||||
|                   data-bs-target="#navs-top-home" |                   data-bs-target="#navs-top-home" | ||||||
| @ -133,9 +241,7 @@ const AttendancePage = () => { | |||||||
|               <li className="nav-item"> |               <li className="nav-item"> | ||||||
|                 <button |                 <button | ||||||
|                   type="button" |                   type="button" | ||||||
|                 className={`nav-link ${ |                   className={`nav-link ${activeTab === "logs" ? "active" : ""} fs-6`} | ||||||
|                   activeTab === "logs" ? "active" : "" |  | ||||||
|                 } fs-6`} |  | ||||||
|                   onClick={() => setActiveTab("logs")} |                   onClick={() => setActiveTab("logs")} | ||||||
|                   data-bs-toggle="tab" |                   data-bs-toggle="tab" | ||||||
|                   data-bs-target="#navs-top-profile" |                   data-bs-target="#navs-top-profile" | ||||||
| @ -156,28 +262,65 @@ const AttendancePage = () => { | |||||||
|                   Regularization |                   Regularization | ||||||
|                 </button> |                 </button> | ||||||
|               </li> |               </li> | ||||||
|  |             </div> | ||||||
|  |             <div className="p-2"> | ||||||
|  |               <input | ||||||
|  |                 type="text" | ||||||
|  |                 className="form-control form-control-sm" | ||||||
|  |                 placeholder="Search employee..." | ||||||
|  |                 value={searchQuery} | ||||||
|  |                 onChange={(e) => setSearchQuery(e.target.value)} | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|           </ul> |           </ul> | ||||||
|           <div className="tab-content attedanceTabs py-0 px-1 px-sm-3"> |           <div className="tab-content attedanceTabs py-0 px-1 px-sm-3"> | ||||||
|             {activeTab === "all" && ( |             {activeTab === "all" && ( | ||||||
|  |               <> | ||||||
|  |                 {!attLoading && ( | ||||||
|                   <div className="tab-pane fade show active py-0"> |                   <div className="tab-pane fade show active py-0"> | ||||||
|                     <Attendance |                     <Attendance | ||||||
|  |                       attendance={filteredAndSearchedTodayAttendance()} | ||||||
|                       handleModalData={handleModalData} |                       handleModalData={handleModalData} | ||||||
|                       getRole={getRole} |                       getRole={getRole} | ||||||
|  |                       setshowOnlyCheckout={setShowPending} | ||||||
|  |                       showOnlyCheckout={showPending} | ||||||
|  |                       searchQuery={searchQuery} | ||||||
|                     /> |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                 )} |                 )} | ||||||
|  |                 {!attLoading && filteredAndSearchedTodayAttendance()?.length === 0 && ( | ||||||
|  |                   <p> | ||||||
|  |                     {" "} | ||||||
|  |                     {showPending | ||||||
|  |                       ? "No Pending Available" | ||||||
|  |                       : "No Employee assigned yet."}{" "} | ||||||
|  |                   </p> | ||||||
|  |                 )} | ||||||
|  |               </> | ||||||
|  |             )} | ||||||
|  | 
 | ||||||
|             {activeTab === "logs" && ( |             {activeTab === "logs" && ( | ||||||
|               <div className="tab-pane fade show active py-0"> |               <div className="tab-pane fade show active py-0"> | ||||||
|                 <AttendanceLog |                 <AttendanceLog | ||||||
|                   handleModalData={handleModalData} |                   handleModalData={handleModalData} | ||||||
|  |                   projectId={selectedProject} | ||||||
|  |                   setshowOnlyCheckout={setShowPending} | ||||||
|  |                   showOnlyCheckout={showPending} | ||||||
|  |                   searchQuery={searchQuery} | ||||||
|                 /> |                 /> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|  | 
 | ||||||
|             {activeTab === "regularization" && DoRegularized && ( |             {activeTab === "regularization" && DoRegularized && ( | ||||||
|               <div className="tab-pane fade show active py-0"> |               <div className="tab-pane fade show active py-0"> | ||||||
|                 <Regularization /> |                 <Regularization | ||||||
|  |                   handleRequest={handleSubmit} | ||||||
|  |                   searchQuery={searchQuery} | ||||||
|  |                 /> | ||||||
|               </div> |               </div> | ||||||
|             )} |             )} | ||||||
|  | 
 | ||||||
|  |             {!attLoading && !attendances && <span>Not Found</span>} | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|  | |||||||
| @ -257,7 +257,7 @@ const DirectoryPageHeader = ({ | |||||||
|                     <ul className="nav nav-tabs mb-0" role="tablist"> |                     <ul className="nav nav-tabs mb-0" role="tablist"> | ||||||
|                         <li className="nav-item" role="presentation"> |                         <li className="nav-item" role="presentation"> | ||||||
|                             <button |                             <button | ||||||
|                                 className={`nav-link ${viewType === "notes" ? "active" : ""}`} |                                 className={`nav-link ${viewType === "notes" ? "active" : ""} fs-6`} | ||||||
|                                 onClick={() => setViewType("notes")} |                                 onClick={() => setViewType("notes")} | ||||||
|                                 type="button" |                                 type="button" | ||||||
|                             > |                             > | ||||||
| @ -266,8 +266,9 @@ const DirectoryPageHeader = ({ | |||||||
|                         </li> |                         </li> | ||||||
|                         <li className="nav-item" role="presentation"> |                         <li className="nav-item" role="presentation"> | ||||||
|                             <button |                             <button | ||||||
|                                 className={`nav-link ${viewType === "card" ? "active" : ""}`} |                                 // Corrected: Apply 'active' if viewType is either 'card' or 'list' | ||||||
|                                 onClick={() => setViewType("card")} |                                 className={`nav-link ${viewType === "card" || viewType === "list" ? "active" : ""} fs-6`} | ||||||
|  |                                 onClick={() => setViewType("card")} // You might want to default to 'card' when switching to contacts | ||||||
|                                 type="button" |                                 type="button" | ||||||
|                             > |                             > | ||||||
|                                 <i className="bx bx-user me-1"></i> Contacts |                                 <i className="bx bx-user me-1"></i> Contacts | ||||||
| @ -279,7 +280,7 @@ const DirectoryPageHeader = ({ | |||||||
|             <hr className="my-0 mb-2" style={{ borderTop: "1px solid #dee2e6" }} /> |             <hr className="my-0 mb-2" style={{ borderTop: "1px solid #dee2e6" }} /> | ||||||
| 
 | 
 | ||||||
|             <div className="row mx-0 px-0 align-items-center mt-2"> |             <div className="row mx-0 px-0 align-items-center mt-2"> | ||||||
|                 <div className="col-12 col-md-6 mb-2 px-5 d-flex align-items-center gap-4"> |                 <div className="col-12 col-md-6 mb-2 px-1 d-flex align-items-center gap-2"> | ||||||
| 
 | 
 | ||||||
|                     <input |                     <input | ||||||
|                         type="search" |                         type="search" | ||||||
| @ -298,7 +299,7 @@ const DirectoryPageHeader = ({ | |||||||
|                                 data-bs-toggle="dropdown" |                                 data-bs-toggle="dropdown" | ||||||
|                                 aria-expanded="false" |                                 aria-expanded="false" | ||||||
|                             > |                             > | ||||||
|                                 <i className={`fa-solid fa-filter ms-1 fs-5 ${notesFilterCount > 0 ? "text-primary" : "text-muted"}`}></i> |                                 <i className={`bx bx-slider-alt ${notesFilterCount > 0 ? "text-primary" : "text-muted"}`}></i> | ||||||
|                                 {notesFilterCount > 0 && ( |                                 {notesFilterCount > 0 && ( | ||||||
|                                     <span className="position-absolute top-0 start-100 translate-middle badge rounded-pill bg-warning" style={{ fontSize: "0.4rem" }}> |                                     <span className="position-absolute top-0 start-100 translate-middle badge rounded-pill bg-warning" style={{ fontSize: "0.4rem" }}> | ||||||
|                                         {notesFilterCount} |                                         {notesFilterCount} | ||||||
| @ -404,17 +405,17 @@ const DirectoryPageHeader = ({ | |||||||
| 
 | 
 | ||||||
|                             <button |                             <button | ||||||
|                                 type="button" |                                 type="button" | ||||||
|                                 className={`btn btn-xs ${viewType === "card" ? "btn-primary" : "btn-outline-primary"}`} |                                 className={`btn  btn-sm p-1 ${viewType === "card" ? "btn-primary" : "btn-outline-primary"}`} | ||||||
|                                 onClick={() => setViewType("card")} |                                 onClick={() => setViewType("card")} | ||||||
|                             > |                             > | ||||||
|                                 <i className="bx bx-grid-alt"></i> |                                 <i className="bx bx-grid-alt"></i> | ||||||
|                             </button> |                             </button> | ||||||
|                             <button |                             <button | ||||||
|                                 type="button" |                                 type="button" | ||||||
|                                 className={`btn btn-xs ${viewType === "list" ? "btn-primary" : "btn-outline-primary"}`} |                                 className={`btn  btn-sm p-1 ${viewType === "list" ? "btn-primary" : "btn-outline-primary"}`} | ||||||
|                                 onClick={() => setViewType("list")} |                                 onClick={() => setViewType("list")} | ||||||
|                             > |                             > | ||||||
|                                 <i className="bx bx-list-ul me-1"></i> |                                 <i className="bx bx-list-ul"></i> | ||||||
| 
 | 
 | ||||||
|                             </button> |                             </button> | ||||||
|                         </div> |                         </div> | ||||||
| @ -422,13 +423,13 @@ const DirectoryPageHeader = ({ | |||||||
| 
 | 
 | ||||||
|                     {/* Filter by funnel icon for Contacts view (retains numerical badge) */} |                     {/* Filter by funnel icon for Contacts view (retains numerical badge) */} | ||||||
|                     {viewType !== "notes" && ( |                     {viewType !== "notes" && ( | ||||||
|                         <div className="dropdown-center" style={{ width: "fit-content" }}> |                         <div className="dropdown" style={{ width: "fit-content" }}> | ||||||
|                             <a |                             <a | ||||||
|                                 className="dropdown-toggle hide-arrow cursor-pointer d-flex align-items-center position-relative" |                                 className="dropdown-toggle hide-arrow cursor-pointer d-flex align-items-center position-relative" | ||||||
|                                 data-bs-toggle="dropdown" |                                 data-bs-toggle="dropdown" | ||||||
|                                 aria-expanded="false" |                                 aria-expanded="false" | ||||||
|                             > |                             > | ||||||
|                                 <i className={`fa-solid fa-filter ms-1 fs-5 ${filtered > 0 ? "text-primary" : "text-muted"}`}></i> |                                 <i className={`bx bx-slider-alt ${filtered > 0 ? "text-primary" : "text-muted"}`}></i> | ||||||
|                                 {filtered > 0 && ( |                                 {filtered > 0 && ( | ||||||
|                                     <span className="position-absolute top-0 start-100 translate-middle badge rounded-pill bg-warning" style={{ fontSize: "0.4rem" }}> |                                     <span className="position-absolute top-0 start-100 translate-middle badge rounded-pill bg-warning" style={{ fontSize: "0.4rem" }}> | ||||||
|                                         {filtered} |                                         {filtered} | ||||||
|  | |||||||
| @ -320,7 +320,7 @@ useEffect(() => { | |||||||
|           {isFilterPanelOpen ? ( |           {isFilterPanelOpen ? ( | ||||||
|             <i className="fa-solid fa-times fs-5" /> |             <i className="fa-solid fa-times fs-5" /> | ||||||
|           ) : ( |           ) : ( | ||||||
|             <i className="fa-solid fa-filter ms-1 fs-5" /> |             <i className="bx bx-slider-alt ms-1" /> | ||||||
|           )} |           )} | ||||||
|         </button> |         </button> | ||||||
|         <div className="activity-section"> |         <div className="activity-section"> | ||||||
| @ -394,7 +394,7 @@ useEffect(() => { | |||||||
|                     > |                     > | ||||||
|                       {batch.documents.map((d, i) => { |                       {batch.documents.map((d, i) => { | ||||||
|                         const hoverDate = moment(d.uploadedAt).format( |                         const hoverDate = moment(d.uploadedAt).format( | ||||||
|                           "DD-MM-YYYY" |                           "DD MMMM, YYYY" | ||||||
|                         ); |                         ); | ||||||
|                         const hoverTime = moment(d.uploadedAt).format( |                         const hoverTime = moment(d.uploadedAt).format( | ||||||
|                           "hh:mm A" |                           "hh:mm A" | ||||||
|  | |||||||
| @ -480,7 +480,7 @@ const EmployeeList = () => { | |||||||
|                           aria-label="User: activate to sort column ascending" |                           aria-label="User: activate to sort column ascending" | ||||||
|                           aria-sort="descending" |                           aria-sort="descending" | ||||||
|                         > |                         > | ||||||
|                           <div className="text-start ms-5">Role</div> |                           <div className="text-start ms-5">Official Designation</div> | ||||||
|                         </th> |                         </th> | ||||||
| 
 | 
 | ||||||
|                         <th |                         <th | ||||||
|  | |||||||
| @ -146,7 +146,7 @@ const EmployeeProfile = () => { | |||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr> |                               </tr> | ||||||
|                               <tr> |                               <tr> | ||||||
|                                 <td className="fw-medium text-start"> |                                 <td className="fw-medium text-start text-nowrap"> | ||||||
|                                   Phone Number: |                                   Phone Number: | ||||||
|                                 </td> |                                 </td> | ||||||
|                                 <td className="text-start"> |                                 <td className="text-start"> | ||||||
| @ -154,26 +154,21 @@ const EmployeeProfile = () => { | |||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr>  |                               </tr>  | ||||||
|                               <tr> |                               <tr> | ||||||
|                                 <td className="fw-medium text-start"> |                                 <td className="fw-medium text-start" style={{ width: '120px' }}> | ||||||
|                                   Emergency Contact Person: |                                   Emergency Contact Person: | ||||||
|                                 </td> |                                 </td> | ||||||
|                                 <td className="text-start"> |                                 <td className="text-start align-bottom"> | ||||||
|                                   {currentEmployee?.emergencyContactPerson || ( |                                   {currentEmployee?.emergencyContactPerson || <em>NA</em>} | ||||||
|                                     <em>NA</em> |  | ||||||
|                                   )} |  | ||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr> |                               </tr> | ||||||
|                               <tr> |                               <tr> | ||||||
|                                 <td className="fw-medium text-start"> |                                 <td className="fw-medium text-start"> | ||||||
|                                   Emergency Contact Number: |                                   Emergency Contact Number: | ||||||
|                                 </td> |                                 </td> | ||||||
|                                 <td className="text-start"> |                                 <td className="text-start align-bottom"> | ||||||
|                                   {currentEmployee?.emergencyPhoneNumber || ( |                                   {currentEmployee?.emergencyPhoneNumber || <em>NA</em>} | ||||||
|                                     <em>NA</em> |  | ||||||
|                                   )} |  | ||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr> |                               </tr> | ||||||
| 
 |  | ||||||
|                               <tr> |                               <tr> | ||||||
|                                 <td className="fw-medium text-start"> |                                 <td className="fw-medium text-start"> | ||||||
|                                   Gender: |                                   Gender: | ||||||
| @ -220,15 +215,14 @@ const EmployeeProfile = () => { | |||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr> |                               </tr> | ||||||
|                               <tr> |                               <tr> | ||||||
|                                 <td className="fw-medium text-start"> |                                 <td className="fw-medium text-start align-top" > | ||||||
|                                   Address: |                                   Address: | ||||||
|                                 </td> |                                 </td> | ||||||
|                                 <td className="text-start"> |                                 <td className="text-start"> | ||||||
|                                   {currentEmployee?.currentAddress || ( |                                   {currentEmployee?.currentAddress || <em>NA</em>} | ||||||
|                                     <em>NA</em> |  | ||||||
|                                   )} |  | ||||||
|                                 </td> |                                 </td> | ||||||
|                               </tr> |                               </tr> | ||||||
|  | 
 | ||||||
|                             </tbody> |                             </tbody> | ||||||
|                           </table> |                           </table> | ||||||
|                         </div> |                         </div> | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ import { | |||||||
|   cacheData, |   cacheData, | ||||||
|   clearCacheKey, |   clearCacheKey, | ||||||
|   getCachedData, |   getCachedData, | ||||||
|  |   useSelectedproject, | ||||||
| } from "../../slices/apiDataManager"; | } from "../../slices/apiDataManager"; | ||||||
| import "./ProjectDetails.css"; | import "./ProjectDetails.css"; | ||||||
| import { | import { | ||||||
| @ -28,8 +29,7 @@ import { setProjectId } from "../../slices/localVariablesSlice"; | |||||||
| 
 | 
 | ||||||
| const ProjectDetails = () => { | const ProjectDetails = () => { | ||||||
| 
 | 
 | ||||||
| 
 |   const projectId = useSelectedproject() | ||||||
|   const projectId = useSelector((store) => store.localVariables.projectId); |  | ||||||
| 
 | 
 | ||||||
|   const { projectNames, fetchData } = useProjectName(); |   const { projectNames, fetchData } = useProjectName(); | ||||||
|   const dispatch = useDispatch() |   const dispatch = useDispatch() | ||||||
| @ -47,9 +47,10 @@ const ProjectDetails = () => { | |||||||
|     refetch, |     refetch, | ||||||
|   } = useProjectDetails(projectId); |   } = useProjectDetails(projectId); | ||||||
| 
 | 
 | ||||||
|   const [activePill, setActivePill] = useState("profile"); |   // const [activePill, setActivePill] = useState("profile"); | ||||||
| 
 |   const [activePill, setActivePill] = useState(() => { | ||||||
| 
 |   return localStorage.getItem("lastActiveProjectTab") || "profile"; | ||||||
|  | }); | ||||||
| 
 | 
 | ||||||
|   const handler = useCallback( |   const handler = useCallback( | ||||||
|     (msg) => { |     (msg) => { | ||||||
| @ -67,8 +68,10 @@ const ProjectDetails = () => { | |||||||
| 
 | 
 | ||||||
|  const handlePillClick = (pillKey) => { |  const handlePillClick = (pillKey) => { | ||||||
|   setActivePill(pillKey); |   setActivePill(pillKey); | ||||||
|  |   localStorage.setItem("lastActiveProjectTab", pillKey); // ✅ Save to localStorage | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|   const renderContent = () => { |   const renderContent = () => { | ||||||
|     if (projectLoading || !projects_Details) return <Loader />; |     if (projectLoading || !projects_Details) return <Loader />; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -192,7 +192,7 @@ const ProjectList = () => { | |||||||
|                   </button> |                   </button> | ||||||
|                 </div> |                 </div> | ||||||
| 
 | 
 | ||||||
|                 <div className="dropdown ms-3 mt-1"> |                  <div className="dropdown mt-1"> | ||||||
|                   <a |                   <a | ||||||
|                     className="dropdown-toggle hide-arrow cursor-pointer p-1 mt-3 " |                     className="dropdown-toggle hide-arrow cursor-pointer p-1 mt-3 " | ||||||
|                     data-bs-toggle="dropdown" |                     data-bs-toggle="dropdown" | ||||||
| @ -200,7 +200,7 @@ const ProjectList = () => { | |||||||
|                     data-bs-custom-class="tooltip" |                     data-bs-custom-class="tooltip" | ||||||
|                     title="Filter" |                     title="Filter" | ||||||
|                   > |                   > | ||||||
|                     <i className="fa-solid fa-filter fs-4"></i> |                     <i className="bx bx-slider-alt ms-1"></i> | ||||||
|                   </a> |                   </a> | ||||||
|                   <ul className="dropdown-menu p-2 text-capitalize"> |                   <ul className="dropdown-menu p-2 text-capitalize"> | ||||||
|                     {[ |                     {[ | ||||||
| @ -269,10 +269,10 @@ const ProjectList = () => { | |||||||
|           <div className="card cursor-pointer"> |           <div className="card cursor-pointer"> | ||||||
|             <div className="card-body p-2"> |             <div className="card-body p-2"> | ||||||
|               <div |               <div | ||||||
|                 className="table-responsive text-nowrap py-2 " |                 className="table-responsive text-nowrap py-2 mx-2" | ||||||
|                 style={{ minHeight: "400px" }} |                 style={{ minHeight: "200px" }} | ||||||
|               > |               > | ||||||
|                 <table className="table m-3"> |                 <table className="table m-0"> | ||||||
|                   <thead> |                   <thead> | ||||||
|                     <tr> |                     <tr> | ||||||
|                       <th className="text-start" colSpan={5}> |                       <th className="text-start" colSpan={5}> | ||||||
|  | |||||||
| @ -2,6 +2,7 @@ import { api } from "../utils/axiosClient"; | |||||||
| 
 | 
 | ||||||
| export const DirectoryRepository = { | export const DirectoryRepository = { | ||||||
|   GetOrganizations: () => api.get("/api/directory/organization"), |   GetOrganizations: () => api.get("/api/directory/organization"), | ||||||
|  |   GetDesignations: () => api.get("/api/directory/designations"), | ||||||
|   GetContacts: (isActive, projectId) => { |   GetContacts: (isActive, projectId) => { | ||||||
|     const params = new URLSearchParams(); |     const params = new URLSearchParams(); | ||||||
|     params.append("active", isActive); |     params.append("active", isActive); | ||||||
|  | |||||||
| @ -5,6 +5,7 @@ import { | |||||||
|   flushApiCache, |   flushApiCache, | ||||||
| } from "../slices/apiCacheSlice"; | } from "../slices/apiCacheSlice"; | ||||||
| import {setLoginUserPermmisions} from "./globalVariablesSlice"; | import {setLoginUserPermmisions} from "./globalVariablesSlice"; | ||||||
|  | import { useSelector } from "react-redux"; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| // Cache data | // Cache data | ||||||
| @ -37,3 +38,16 @@ export const cacheProfileData = ( data) => { | |||||||
| export const getCachedProfileData = () => { | export const getCachedProfileData = () => { | ||||||
|   return store.getState().globalVariables.loginUser; |   return store.getState().globalVariables.loginUser; | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | export const useSelectedproject = () => { | ||||||
|  |   const selectedProject = useSelector((store)=> store.localVariables.projectId); | ||||||
|  |   var project = localStorage.getItem("project"); | ||||||
|  |   if(project){ | ||||||
|  |     return project | ||||||
|  |   } else{ | ||||||
|  |     return selectedProject | ||||||
|  |   } | ||||||
|  |   // return project ? selectedProject | ||||||
|  |    | ||||||
|  |    | ||||||
|  | };  | ||||||
| @ -21,7 +21,9 @@ const localVariablesSlice = createSlice({ | |||||||
|       state.regularizationCount = action.payload; |       state.regularizationCount = action.payload; | ||||||
|     }, |     }, | ||||||
|     setProjectId: (state, action) => {  |     setProjectId: (state, action) => {  | ||||||
|  |       localStorage.setItem("project",null) | ||||||
|       state.projectId = action.payload; |       state.projectId = action.payload; | ||||||
|  |       localStorage.setItem("project",state.projectId) | ||||||
|     }, |     }, | ||||||
|     refreshData: ( state, action ) => |     refreshData: ( state, action ) => | ||||||
|     { |     { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user