added status and search filter filter
This commit is contained in:
		
							parent
							
								
									c61cb3bb63
								
							
						
					
					
						commit
						3ee5a71cc9
					
				| @ -113,7 +113,9 @@ const ProjectList = () => { | |||||||
|   const totalPages = Math.ceil(filteredProjects.length / itemsPerPage); |   const totalPages = Math.ceil(filteredProjects.length / itemsPerPage); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|       const tooltipTriggerList = Array.from(document.querySelectorAll('[data-bs-toggle="tooltip"]')); |     const tooltipTriggerList = Array.from( | ||||||
|  |       document.querySelectorAll('[data-bs-toggle="tooltip"]') | ||||||
|  |     ); | ||||||
|     tooltipTriggerList.forEach((el) => new bootstrap.Tooltip(el)); |     tooltipTriggerList.forEach((el) => new bootstrap.Tooltip(el)); | ||||||
|   }, []); |   }, []); | ||||||
| 
 | 
 | ||||||
| @ -163,7 +165,11 @@ const ProjectList = () => { | |||||||
|                   listView ? "btn-primary" : "btn-outline-primary" |                   listView ? "btn-primary" : "btn-outline-primary" | ||||||
|                 }`} |                 }`} | ||||||
|                 onClick={() => setListView(true)} |                 onClick={() => setListView(true)} | ||||||
|                 data-bs-toggle="tooltip" data-bs-offset="0,8" data-bs-placement="top" data-bs-custom-class="tooltip" title="List View" |                 data-bs-toggle="tooltip" | ||||||
|  |                 data-bs-offset="0,8" | ||||||
|  |                 data-bs-placement="top" | ||||||
|  |                 data-bs-custom-class="tooltip" | ||||||
|  |                 title="List View" | ||||||
|               > |               > | ||||||
|                 <i className="bx bx-list-ul"></i> |                 <i className="bx bx-list-ul"></i> | ||||||
|               </button> |               </button> | ||||||
| @ -174,7 +180,11 @@ const ProjectList = () => { | |||||||
|                   !listView ? "btn-primary" : "btn-outline-primary" |                   !listView ? "btn-primary" : "btn-outline-primary" | ||||||
|                 }`} |                 }`} | ||||||
|                 onClick={() => setListView(false)} |                 onClick={() => setListView(false)} | ||||||
|                 data-bs-toggle="tooltip" data-bs-offset="0,8" data-bs-placement="top" data-bs-custom-class="tooltip" title="Card View" |                 data-bs-toggle="tooltip" | ||||||
|  |                 data-bs-offset="0,8" | ||||||
|  |                 data-bs-placement="top" | ||||||
|  |                 data-bs-custom-class="tooltip" | ||||||
|  |                 title="Card View" | ||||||
|               > |               > | ||||||
|                 <i className="bx bx-grid-alt"></i> |                 <i className="bx bx-grid-alt"></i> | ||||||
|               </button> |               </button> | ||||||
| @ -202,10 +212,75 @@ const ProjectList = () => { | |||||||
| 
 | 
 | ||||||
|         <div className="row"> |         <div className="row"> | ||||||
|           {listView ? ( |           {listView ? ( | ||||||
|             <ProjectListView |             <div className="table-responsive text-nowrap py-2 "> | ||||||
|               projectsData={currentItems} |               <table className="table px-2"> | ||||||
|               onStatusFilterChange={handleStatusFilterFromChild} |                 <thead> | ||||||
|  |                   <tr> | ||||||
|  |                     <th className="text-start" colSpan={5}> | ||||||
|  |                       Project Name | ||||||
|  |                     </th> | ||||||
|  |                     <th className="mx-2">Project Manger</th> | ||||||
|  |                     <th className="mx-2">START DATE</th> | ||||||
|  |                     <th className="mx-2">DEADLINE</th> | ||||||
|  |                     <th className="mx-2">Task</th> | ||||||
|  |                     <th className="mx-2">Progress</th> | ||||||
|  |                     <th className="mx-2"> | ||||||
|  |                       <div className="dropdown"> | ||||||
|  |                         <a | ||||||
|  |                           className="dropdown-toggle hide-arrow cursor-pointer" | ||||||
|  |                           data-bs-toggle="dropdown" | ||||||
|  |                           aria-expanded="false" | ||||||
|  |                         > | ||||||
|  |                           Status <i className="bx bx-filter"></i> | ||||||
|  |                         </a> | ||||||
|  |                         <ul className="dropdown-menu p-2 text-capitalize"> | ||||||
|  |                           {[ | ||||||
|  |                             { id: 1, label: "Active" }, | ||||||
|  |                             { id: 2, label: "On Hold" }, | ||||||
|  |                             { id: 3, label: "Inactive" }, | ||||||
|  |                             { id: 4, label: "Completed" }, | ||||||
|  |                           ].map(({ id, label }) => ( | ||||||
|  |                             <li key={id}> | ||||||
|  |                               <div className="form-check"> | ||||||
|  |                                 <input | ||||||
|  |                                   className="form-check-input  " | ||||||
|  |                                   type="checkbox" | ||||||
|  |                                   checked={selectedStatuses.includes(id)} | ||||||
|  |                                   onChange={() => handleStatusChange(id)} | ||||||
|                                 /> |                                 /> | ||||||
|  |                                 <label className="form-check-label"> | ||||||
|  |                                   {label} | ||||||
|  |                                 </label> | ||||||
|  |                               </div> | ||||||
|  |                             </li> | ||||||
|  |                           ))} | ||||||
|  |                         </ul> | ||||||
|  |                       </div> | ||||||
|  |                     </th> | ||||||
|  |                     <th | ||||||
|  |                       className={`mx-2 ${ | ||||||
|  |                         HasManageProject ? "d-sm-table-cell" : "d-none" | ||||||
|  |                       }`} | ||||||
|  |                     > | ||||||
|  |                       Action | ||||||
|  |                     </th> | ||||||
|  |                   </tr> | ||||||
|  |                 </thead> | ||||||
|  |                 <tbody className="table-border-bottom-0 overflow-auto "> | ||||||
|  |                   {currentItems.length === 0 ? ( | ||||||
|  |                     <tr> | ||||||
|  |                       <td colSpan="12" className="text-center py-4"> | ||||||
|  |                         No projects found | ||||||
|  |                       </td> | ||||||
|  |                     </tr> | ||||||
|  |                   ) : ( | ||||||
|  |                     currentItems.map((project) => ( | ||||||
|  |                       <ProjectListView key={project.id} projectData={project} /> | ||||||
|  |                     )) | ||||||
|  |                   )} | ||||||
|  |                 </tbody> | ||||||
|  |               </table> | ||||||
|  |             </div> | ||||||
|           ) : ( |           ) : ( | ||||||
|             currentItems.map((project) => ( |             currentItems.map((project) => ( | ||||||
|               <ProjectCard key={project.id} projectData={project} /> |               <ProjectCard key={project.id} projectData={project} /> | ||||||
|  | |||||||
| @ -1,7 +1,10 @@ | |||||||
| import React, { useState, useEffect } from "react"; | import React, { useState, useEffect } from "react"; | ||||||
| import moment from "moment"; | import moment from "moment"; | ||||||
| import { useProjects } from "../../hooks/useProjects"; | import { useProjects } from "../../hooks/useProjects"; | ||||||
| import { getProjectStatusName,getProjectStatusColor } from "../../utils/projectStatus"; | import { | ||||||
|  |   getProjectStatusName, | ||||||
|  |   getProjectStatusColor, | ||||||
|  | } from "../../utils/projectStatus"; | ||||||
| import ProgressBar from "../../components/common/ProgressBar"; | import ProgressBar from "../../components/common/ProgressBar"; | ||||||
| import { useNavigate } from "react-router-dom"; | import { useNavigate } from "react-router-dom"; | ||||||
| import ManageProject from "../../components/Project/ManageProject"; | import ManageProject from "../../components/Project/ManageProject"; | ||||||
| @ -12,44 +15,17 @@ import ManageProjectInfo from "../../components/Project/ManageProjectInfo"; | |||||||
| import showToast from "../../services/toastService"; | import showToast from "../../services/toastService"; | ||||||
| import { getCachedData, cacheData } from "../../slices/apiDataManager"; | import { getCachedData, cacheData } from "../../slices/apiDataManager"; | ||||||
| 
 | 
 | ||||||
| const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | const ProjectListView = ({ projectData }) => { | ||||||
|  |   const [projectInfo, setProjectInfo] = useState(projectData); | ||||||
|   const [projectDetails, setProjectDetails] = useState(null); |   const [projectDetails, setProjectDetails] = useState(null); | ||||||
|   const[projectInfo,setProjectInfo] = useState(null) |  | ||||||
|   const [showModal, setShowModal] = useState(false); |   const [showModal, setShowModal] = useState(false); | ||||||
|   const projects = Array.isArray(projectsData) ? projectsData : []; |   const navigate = useNavigate(); | ||||||
|  |   const ManageProject = useHasUserPermission(MANAGE_PROJECT); | ||||||
| 
 | 
 | ||||||
|   const [ selectedStatuses, setSelectedStatuses ] = useState( [ 1, 2, 3, 4 ] ); |   const handleShow = async () => { | ||||||
|   const navigate = useNavigate() |  | ||||||
| 
 |  | ||||||
|   const handleStatusChange = (statusId) => { |  | ||||||
|     const updated = selectedStatuses.includes(statusId) |  | ||||||
|       ? selectedStatuses.filter((id) => id !== statusId) |  | ||||||
|       : [...selectedStatuses, statusId]; |  | ||||||
| 
 |  | ||||||
|     setSelectedStatuses(updated); |  | ||||||
|     onStatusFilterChange(updated);  |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   useEffect(() => { |  | ||||||
|     onStatusFilterChange(selectedStatuses); |  | ||||||
|   }, []); |  | ||||||
| 
 |  | ||||||
|   const getProgress = (planned, completed) => { |  | ||||||
|     return (completed * 100) / planned + "%"; |  | ||||||
|   }; |  | ||||||
|   const getProgressInNumber = (planned, completed) => { |  | ||||||
|     return (completed * 100) / planned; |  | ||||||
|   }; |  | ||||||
|   const filteredProjects = projects.filter((project) => |  | ||||||
|     selectedStatuses.includes(project.projectStatusId) |  | ||||||
|   ); |  | ||||||
| 
 |  | ||||||
|   const handleShow = async ( project) => |  | ||||||
|   { |  | ||||||
|     setProjectInfo(project) |  | ||||||
|     try { |     try { | ||||||
|       const response = await ProjectRepository.getProjectByprojectId( |       const response = await ProjectRepository.getProjectByprojectId( | ||||||
|         project.id |         projectInfo.id | ||||||
|       ); |       ); | ||||||
|       setProjectDetails(response.data); |       setProjectDetails(response.data); | ||||||
|       setShowModal(true); |       setShowModal(true); | ||||||
| @ -57,8 +33,22 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|       showToast("Failed to load project details", "error"); |       showToast("Failed to load project details", "error"); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|  |   const getProgress = (planned, completed) => { | ||||||
|  |     return (completed * 100) / planned + "%"; | ||||||
|  |   }; | ||||||
|  |   const getProgressInNumber = (planned, completed) => { | ||||||
|  |     return (completed * 100) / planned; | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   const handleClose = () => setShowModal(false); |   const handleClose = () => setShowModal(false); | ||||||
|   const handleFormSubmit = (updatedProject,projectInfo) => { | 
 | ||||||
|  | 
 | ||||||
|  |   const handleViewProject = () => { | ||||||
|  |     navigate(`/projects/${projectData.id}`); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const handleFormSubmit = (updatedProject) => { | ||||||
|     if (projectInfo?.id) { |     if (projectInfo?.id) { | ||||||
|       ProjectRepository.updateProject(projectInfo.id, updatedProject) |       ProjectRepository.updateProject(projectInfo.id, updatedProject) | ||||||
|         .then((response) => { |         .then((response) => { | ||||||
| @ -68,7 +58,7 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|             building: projectDetails?.building, |             building: projectDetails?.building, | ||||||
|           }; |           }; | ||||||
| 
 | 
 | ||||||
|           setProjectInfo(updatedProject); |           setProjectInfo( updatedProjectData ); | ||||||
| 
 | 
 | ||||||
|           if (getCachedData(`projectinfo-${projectInfo.id}`)) { |           if (getCachedData(`projectinfo-${projectInfo.id}`)) { | ||||||
|             cacheData(`projectinfo-${projectInfo.id}`, updatedProjectData); |             cacheData(`projectinfo-${projectInfo.id}`, updatedProjectData); | ||||||
| @ -84,7 +74,6 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|             cacheData("projectslist", updatedProjectsList); |             cacheData("projectslist", updatedProjectsList); | ||||||
|           } |           } | ||||||
|           showToast("Project updated successfully.", "success"); |           showToast("Project updated successfully.", "success"); | ||||||
| 
 |  | ||||||
|           setShowModal(false); |           setShowModal(false); | ||||||
|         }) |         }) | ||||||
|         .catch((error) => { |         .catch((error) => { | ||||||
| @ -92,9 +81,8 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 |   return ( | ||||||
|   return (<> |     <> | ||||||
|    |  | ||||||
|       {showModal && projectDetails && ( |       {showModal && projectDetails && ( | ||||||
|         <div |         <div | ||||||
|           className="modal fade show" |           className="modal fade show" | ||||||
| @ -110,100 +98,52 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|           /> |           /> | ||||||
|         </div> |         </div> | ||||||
|       )} |       )} | ||||||
|     <div className="table-responsive text-nowrap py-2 "> |  | ||||||
|       <table className="table px-2"> |  | ||||||
|         <thead> |  | ||||||
|           <tr> |  | ||||||
|             <th className="text-start" colSpan={5}> |  | ||||||
|               Project Name |  | ||||||
|             </th> |  | ||||||
|             <th className="mx-2">Project Manger</th> |  | ||||||
|             <th className="mx-2">START DATE</th> |  | ||||||
|             <th className="mx-2">DEADLINE</th> |  | ||||||
|             <th className="mx-2">Task</th> |  | ||||||
|             <th className="mx-2">Progress</th> |  | ||||||
|             <th className="mx-2"> |  | ||||||
|               <div className="dropdown"> |  | ||||||
|                 <a |  | ||||||
|                   className="dropdown-toggle hide-arrow cursor-pointer" |  | ||||||
|                   data-bs-toggle="dropdown" |  | ||||||
|                   aria-expanded="false" |  | ||||||
|                 > |  | ||||||
|                   Status <i className="bx bx-filter"></i> |  | ||||||
|                 </a> |  | ||||||
|                 <ul className="dropdown-menu p-2 text-capitalize"> |  | ||||||
|                   {[ |  | ||||||
|                     { id: 1, label: "Active" }, |  | ||||||
|                     { id: 2, label: "On Hold" }, |  | ||||||
|                     { id: 3, label: "Inactive" }, |  | ||||||
|                     { id: 4, label: "Completed" }, |  | ||||||
|                   ].map(({ id, label }) => ( |  | ||||||
|                     <li key={id}> |  | ||||||
|                       <div className="form-check"> |  | ||||||
|                         <input |  | ||||||
|                           className="form-check-input  " |  | ||||||
|                           type="checkbox" |  | ||||||
|                           checked={selectedStatuses.includes(id)} |  | ||||||
|                           onChange={() => handleStatusChange( id )} |  | ||||||
| 
 | 
 | ||||||
|                         /> |       <tr className="py-8"> | ||||||
|                         <label className="form-check-label">{label}</label> |  | ||||||
|                       </div> |  | ||||||
|                     </li> |  | ||||||
|                   ))} |  | ||||||
|                 </ul> |  | ||||||
|               </div> |  | ||||||
|             </th> |  | ||||||
|             <th className="mx-2">Action</th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody |  | ||||||
|           className="table-border-bottom-0 overflow-auto " |  | ||||||
|           style={{ maxHeight: "50%", minHeight: "40%" }} |  | ||||||
|         > |  | ||||||
|           {filteredProjects.length === 0 ? ( |  | ||||||
|             <tr> |  | ||||||
|               <td colSpan="12" className="text-center py-4"> |  | ||||||
|                 No projects found  |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           ) : ( |  | ||||||
|             filteredProjects.map((project) => ( |  | ||||||
|               <tr key={project.id} className="py-8"> |  | ||||||
|         <td className="text-start" colSpan={5}> |         <td className="text-start" colSpan={5}> | ||||||
|                   <strong className="text-primary cursor-pointer" onClick={()=>navigate(`/projects/${project.id}`)}>{project.name}</strong> |           <strong | ||||||
|  |             className="text-primary cursor-pointer" | ||||||
|  |             onClick={() => navigate(`/projects/${projectInfo.id}`)} | ||||||
|  |           > | ||||||
|  |             {projectInfo.name} | ||||||
|  |           </strong> | ||||||
|         </td> |         </td> | ||||||
|                 <td className="text-start small">{project.contactPerson}</td> |         <td className="text-start small">{projectInfo.contactPerson}</td> | ||||||
|         <td className="small text-start"> |         <td className="small text-start"> | ||||||
|           <small> |           <small> | ||||||
|                     {project.startDate |             {projectInfo.startDate | ||||||
|                       ? moment(project.startDate).format("DD-MMM-YYYY") |               ? moment(projectInfo.startDate).format("DD-MMM-YYYY") | ||||||
|               : "NA"} |               : "NA"} | ||||||
|           </small> |           </small> | ||||||
|         </td> |         </td> | ||||||
|         <td className="mx-2 text-start small"> |         <td className="mx-2 text-start small"> | ||||||
|                   {project.endDate |           {projectInfo.endDate | ||||||
|                     ? moment(project.endDate).format("DD-MMM-YYYY") |             ? moment(projectInfo.endDate).format("DD-MMM-YYYY") | ||||||
|             : "NA"} |             : "NA"} | ||||||
|         </td> |         </td> | ||||||
|                 <td className="mx-2 text-start small">{project.plannedWork}</td> |         <td className="mx-2 text-start small">{projectInfo.plannedWork}</td> | ||||||
|         <td className="py-6 mx-2 text-start small align-items-center"> |         <td className="py-6 mx-2 text-start small align-items-center"> | ||||||
|                   <ProgressBar plannedWork={project.plannedWork} completedWork={project.completedWork} className="mb-0" height="4px"/> |           <ProgressBar | ||||||
|  |             plannedWork={projectInfo.plannedWork} | ||||||
|  |             completedWork={projectInfo.completedWork} | ||||||
|  |             className="mb-0" | ||||||
|  |             height="4px" | ||||||
|  |           /> | ||||||
|         </td> |         </td> | ||||||
| 
 | 
 | ||||||
|         <td className="mx-6"> |         <td className="mx-6"> | ||||||
|           <p className="mb-0"> |           <p className="mb-0"> | ||||||
|             <span |             <span | ||||||
|               className={`badge ${getProjectStatusColor( |               className={`badge ${getProjectStatusColor( | ||||||
|                         project.projectStatusId |                 projectInfo.projectStatusId | ||||||
|               )}`} |               )}`} | ||||||
|             > |             > | ||||||
|                       {getProjectStatusName(project.projectStatusId)} |               {getProjectStatusName(projectInfo.projectStatusId)} | ||||||
|             </span> |             </span> | ||||||
|           </p> |           </p> | ||||||
|         </td> |         </td> | ||||||
| 
 | 
 | ||||||
|                 <td className="mx-2"> |         <td className={`mx-2 ${ManageProject ? "d-sm-table-cell":"d-none"}`}> | ||||||
|           <div className="dropdown z-2"> |           <div className="dropdown z-2"> | ||||||
|             <button |             <button | ||||||
|               type="button" |               type="button" | ||||||
| @ -225,14 +165,14 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|                 <a |                 <a | ||||||
|                   aria-label="click to  View details" |                   aria-label="click to  View details" | ||||||
|                   className="dropdown-item" |                   className="dropdown-item" | ||||||
|                         onClick={()=> navigate(`/projects/${project.id}`)} |                   onClick={() => navigate(`/projects/${projectInfo.id}`)} | ||||||
|                 > |                 > | ||||||
|                   <i className="bx bx-detail me-2"></i> |                   <i className="bx bx-detail me-2"></i> | ||||||
|                   <span className="align-left">View details</span> |                   <span className="align-left">View details</span> | ||||||
|                 </a> |                 </a> | ||||||
|               </li> |               </li> | ||||||
| 
 | 
 | ||||||
|                     <li  onClick={()=>handleShow(project)}> |               <li onClick={handleShow}> | ||||||
|                 <a className="dropdown-item"> |                 <a className="dropdown-item"> | ||||||
|                   <i className="bx bx-pencil me-2"></i> |                   <i className="bx bx-pencil me-2"></i> | ||||||
|                   <span className="align-left">Modify</span> |                   <span className="align-left">Modify</span> | ||||||
| @ -240,9 +180,7 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|               </li> |               </li> | ||||||
|               <li |               <li | ||||||
|                 onClick={() => |                 onClick={() => | ||||||
|                         navigate( |                   navigate(`/activities/records?project=${projectInfo.id}`) | ||||||
|                           `/activities/records?project=${project.id}` |  | ||||||
|                         ) |  | ||||||
|                 } |                 } | ||||||
|               > |               > | ||||||
|                 <a className="dropdown-item"> |                 <a className="dropdown-item"> | ||||||
| @ -254,13 +192,7 @@ const ProjectListView = ( {projectsData, onStatusFilterChange} ) =>{ | |||||||
|           </div> |           </div> | ||||||
|         </td> |         </td> | ||||||
|       </tr> |       </tr> | ||||||
|             )) |  | ||||||
|           )} |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|     </> |     </> | ||||||
| 
 |  | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Pramod Mahajan
						Pramod Mahajan