-
-
Notifications
You must be signed in to change notification settings - Fork 57
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: new project details page (#1070)
* fix (select): className added to select * feat (routes): newproject_details route added * feat (assetModules): icons added * feat (newProjectDetails): new UI for projectDetails * feat (newProjectDetails): navigate to specified URL on click * fix (select): classname added * fix (activitiesPanel): bg added * fix (select): show title if present * fix (newProjectDetails): new UI width changes * fix (newProjectDetails): height/width adjusted * feat (assetModules): icons added * feat (newProjectDetails): slicing for taskActivity activities section * feat (newProjectDetails): zoomToTask feature added on mapIcon click * fix (newProjectDetails): desc gaps reduced * feat (newProjectDetails): necessary files added to ProjectDetailsV2 folder and imports updated * feat (newProjectDetails): skeletonLoader added to projectDetails section * feat: added new field to user model to store user profile_img * feat: updated schema to add updated task history * fix (newProjectDetails): taskHistorySection: dynamic value for username, userProfilePic, changedToStatus * fix (newProjectDetails): color adjustment * fix (newProjectDetails): taskHistory - changed taskStatus override issue solved * fix (codeRefactor): comments removed * updated response of update_task_status * feat (newProjectDetails): projectDashboard api fetch * fix (enwProjectDetails): textColor according to status changed * feat (newProjectDetails): taskActivity - container added for sorting task activities * fix (newProjectDetails): organization logo fixes * fix (newProjectDetails): UI fix * fix (themeSlice): statusTextColor - text color changed * fix (newProjectDetails): dynamic location added * fix (projectDetails): added new UI button to redirect to the new project details UI --------- Co-authored-by: sujanadh <[email protected]>
- Loading branch information
Showing
17 changed files
with
1,334 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
208 changes: 208 additions & 0 deletions
208
src/frontend/src/components/ProjectDetailsV2/ActivitiesPanel.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,208 @@ | ||
/* eslint-disable react/jsx-key */ | ||
import React, { useEffect, useState } from 'react'; | ||
import environment from '../../environment'; | ||
import CoreModules from '../../shared/CoreModules'; | ||
import AssetModules from '../../shared/AssetModules'; | ||
import { CustomSelect } from '../../components/common/Select'; | ||
import profilePic from '../../assets/images/project_icon.png'; | ||
import { Feature } from 'ol'; | ||
import { Polygon } from 'ol/geom'; | ||
import { ActivitiesCardSkeletonLoader, ShowingCountSkeletonLoader } from './SkeletonLoader'; | ||
|
||
const sortByList = [ | ||
{ id: 'activities', name: 'Activities' }, | ||
{ id: 'date', name: 'Date' }, | ||
{ id: 'users', name: 'Users' }, | ||
{ id: 'taskid', name: 'Task ID' }, | ||
]; | ||
|
||
const ActivitiesPanel = ({ defaultTheme, state, params, map, view, mapDivPostion, states }) => { | ||
const displayLimit = 10; | ||
const [searchText, setSearchText] = useState(''); | ||
const [taskHistories, setTaskHistories] = useState([]); | ||
const [taskDisplay, setTaskDisplay] = React.useState(displayLimit); | ||
const [allActivities, setAllActivities] = useState(0); | ||
const [sortBy, setSortBy] = useState(null); | ||
const [showShortBy, setShowSortBy] = useState(false); | ||
const projectDetailsLoading = CoreModules.useAppSelector((state) => state?.project?.projectDetailsLoading); | ||
|
||
const handleOnchange = (event) => { | ||
setSearchText(event.target.value); | ||
}; | ||
|
||
useEffect(() => { | ||
const index = state.findIndex((project) => project.id == environment.decode(params.id)); | ||
let taskHistories = []; | ||
|
||
if (index != -1) { | ||
state[index].taskBoundries.forEach((task) => { | ||
taskHistories = taskHistories.concat( | ||
task.task_history.map((history) => { | ||
return { | ||
...history, | ||
changedToStatus: history.status, | ||
taskId: task.id, | ||
status: task.task_status, | ||
outlineGeojson: task.outline_geojson, | ||
}; | ||
}), | ||
); | ||
}); | ||
} | ||
setAllActivities(taskHistories.length); | ||
|
||
let finalTaskHistory = taskHistories.filter((task) => { | ||
return ( | ||
task.taskId.toString().includes(searchText) || | ||
task.action_text.split(':')[1].replace(/\s+/g, '').toString().includes(searchText.toString()) | ||
); | ||
}); | ||
|
||
if (searchText != '') { | ||
setTaskHistories(finalTaskHistory); | ||
} else { | ||
setTaskHistories(taskHistories); | ||
} | ||
}, [taskDisplay, state, searchText]); | ||
|
||
const zoomToTask = (taskId) => { | ||
const geojson = taskHistories | ||
.filter((history) => history.taskId === taskId) | ||
.map((history) => history.outlineGeojson)[0]; | ||
|
||
const olFeature = new Feature({ | ||
geometry: new Polygon(geojson.geometry.coordinates).transform('EPSG:4326', 'EPSG:3857'), | ||
}); | ||
// Get the extent of the OpenLayers feature | ||
const extent = olFeature.getGeometry().getExtent(); | ||
map.getView().fit(extent, { | ||
padding: [0, 0, 0, 0], | ||
}); | ||
}; | ||
|
||
const ActivitiesCard = ({ taskHistory }) => { | ||
const actionDate = taskHistory?.action_date?.split('T')[0]; | ||
const actionTime = `${taskHistory?.action_date?.split('T')[1].split(':')[0]}:${taskHistory?.action_date | ||
?.split('T')[1] | ||
.split(':')[1]}`; | ||
return ( | ||
<div className="fmtm-flex fmtm-gap-2 fmtm-items-center fmtm-justify-between fmtm-px-1 fmtm-border-b-[2px] fmtm-border-white fmtm-py-3"> | ||
<div className="fmtm-flex fmtm-items-center"> | ||
<div className="fmtm-w-[2.81rem] fmtm-h-[2.81rem] fmtm-border fmtm-rounded-full fmtm-overflow-hidden fmtm-mr-4"> | ||
{taskHistory?.profile_img ? ( | ||
<img src={taskHistory?.profile_img} alt="Profile Picture" /> | ||
) : ( | ||
<div className="fmtm-w-full fmtm-h-full fmtm-flex fmtm-justify-center fmtm-items-center fmtm-bg-white"> | ||
<AssetModules.PersonIcon color="success" sx={{ fontSize: '30px' }} /> | ||
</div> | ||
)} | ||
</div> | ||
<div className="fmtm-text-base"> | ||
<span className="fmtm-text-[#555555] fmtm-font-medium fmtm-font-archivo">{taskHistory?.username} </span> | ||
<span className="fmtm-text-[#7A7676] fmtm-font-extralight fmtm-italic fmtm-font-archivo"> | ||
updated status to{' '} | ||
</span> | ||
<p | ||
style={{ color: defaultTheme.statusTextTheme[taskHistory?.changedToStatus] }} | ||
className="fmtm-font-archivo" | ||
> | ||
{taskHistory?.changedToStatus} | ||
</p> | ||
<div className="fmtm-flex fmtm-items-center fmtm-justify-between"> | ||
<p className="fmtm-font-archivo fmtm-text-sm fmtm-text-[#7A7676]">#{taskHistory.taskId}</p> | ||
<div className="fmtm-flex fmtm-items-center fmtm-mb-1"> | ||
<AssetModules.AccessTimeIcon className="fmtm-text-primaryRed" style={{ fontSize: '20px' }} /> | ||
</div> | ||
<p className="fmtm-font-archivo fmtm-text-sm fmtm-text-[#7A7676]"> | ||
<span>{actionDate} </span> | ||
<span>{actionTime}</span> | ||
</p> | ||
</div> | ||
</div> | ||
</div> | ||
<div title="Zoom to Task" onClick={() => zoomToTask(taskHistory.taskId)}> | ||
<AssetModules.MapIcon | ||
className="fmtm-text-[#9B9999] hover:fmtm-text-[#555555] fmtm-cursor-pointer" | ||
style={{ fontSize: '20px' }} | ||
/> | ||
</div> | ||
</div> | ||
); | ||
}; | ||
|
||
return ( | ||
<div className="fmtm-w-full fmtm-px-2 sm:fmtm-px-0 fmtm-relative"> | ||
<div className="fmtm-sticky sm:fmtm-relative -fmtm-top-[1px] fmtm-bg-white sm:fmtm-bg-[#F5F5F5]"> | ||
<div className="fmtm-flex fmtm-items-center fmtm-w-full fmtm-justify-between"> | ||
<input | ||
type="text" | ||
onChange={handleOnchange} | ||
value={searchText} | ||
placeholder="Search by task id or username" | ||
className="fmtm-min-w-[14.3rem] fmtm-w-[67%] fmtm-text-md fmtm-px-1 fmtm-py-[0.18rem] fmtm-outline-none fmtm-border-[1px] fmtm-border-[#E7E2E2]" | ||
/> | ||
<div> | ||
<div className="fmtm-relative"> | ||
<div | ||
className="fmtm-flex fmtm-items-center fmtm-gap-1 fmtm-justify-end fmtm-cursor-pointer" | ||
onClick={() => setShowSortBy(!showShortBy)} | ||
> | ||
<AssetModules.ImportExportIcon | ||
className={`${sortBy ? 'fmtm-text-primaryRed' : 'fmtm-text-gray-500'}`} | ||
/> | ||
<p className="fmtm-text-gray-500">Sort</p> | ||
</div> | ||
{showShortBy && ( | ||
<div className="fmtm-w-[10rem] fmtm-absolute fmtm-top-6 fmtm-right-0 sm:fmtm-left-0 fmtm-bg-[#fafafa] sm:fmtm-bg-white fmtm-z-[999] fmtm-rounded-sm fmtm-overflow-hidden"> | ||
{/* <p className="fmtm-border-b-[1px] fmtm-px-2 fmtm-py-1">Sort By:</p> */} | ||
{sortByList.map((item, i) => ( | ||
<div | ||
className={`fmtm-flex fmtm-items-center fmtm-gap-4 fmtm-px-2 fmtm-py-1 hover:fmtm-bg-red-50 hover:fmtm-cursor-pointer fmtm-duration-150 ${ | ||
sortByList.length - 1 > i && | ||
'fmtm-border-b-[1px] fmtm-border-b-slate-200 sm:fmtm-border-gray-100' | ||
}`} | ||
onClick={() => { | ||
if (item.name === sortBy) { | ||
setSortBy(null); | ||
} else { | ||
setSortBy(item.name); | ||
} | ||
setShowSortBy(false); | ||
}} | ||
> | ||
{/* {sortBy === item.name &&} */} | ||
<AssetModules.CheckIcon | ||
className={`${sortBy === item.name ? 'fmtm-visible' : 'fmtm-invisible'} fmtm-text-primaryRed`} | ||
/> | ||
<div>{item.name}</div> | ||
</div> | ||
))} | ||
</div> | ||
)} | ||
</div> | ||
</div> | ||
</div> | ||
{projectDetailsLoading ? ( | ||
<ShowingCountSkeletonLoader /> | ||
) : ( | ||
<p className="fmtm-text-[#A8A6A6] fmtm-text-base fmtm-my-1"> | ||
showing {taskHistories?.length} of {allActivities} activities | ||
</p> | ||
)} | ||
</div> | ||
<div className="sm:fmtm-h-[52vh] fmtm-overflow-y-scroll scrollbar"> | ||
{projectDetailsLoading ? ( | ||
<div> | ||
{Array.from({ length: 10 }).map((i) => ( | ||
<ActivitiesCardSkeletonLoader key={i} /> | ||
))} | ||
</div> | ||
) : ( | ||
<div>{taskHistories?.map((taskHistory) => <ActivitiesCard taskHistory={taskHistory} />)}</div> | ||
)} | ||
</div> | ||
</div> | ||
); | ||
}; | ||
|
||
export default ActivitiesPanel; |
72 changes: 72 additions & 0 deletions
72
src/frontend/src/components/ProjectDetailsV2/MapControlComponent.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,72 @@ | ||
import React, { useState } from 'react'; | ||
import AssetModules from '../../shared/AssetModules'; | ||
import VectorLayer from 'ol/layer/Vector'; | ||
import CoreModules from '../../shared/CoreModules.js'; | ||
import { ProjectActions } from '../../store/slices/ProjectSlice'; | ||
|
||
const MapControlComponent = ({ map }) => { | ||
const btnList = [ | ||
{ | ||
id: 'add', | ||
icon: <AssetModules.AddIcon />, | ||
}, | ||
{ | ||
id: 'minus', | ||
icon: <AssetModules.RemoveIcon />, | ||
}, | ||
{ | ||
id: 'currentLocation', | ||
icon: <AssetModules.MyLocationIcon />, | ||
}, | ||
{ | ||
id: 'taskBoundries', | ||
icon: <AssetModules.CropFreeIcon />, | ||
}, | ||
]; | ||
const dispatch = CoreModules.useAppDispatch(); | ||
const [toggleCurrentLoc, setToggleCurrentLoc] = useState(false); | ||
const geolocationStatus = CoreModules.useAppSelector((state) => state.project.geolocationStatus); | ||
const handleOnClick = (btnId) => { | ||
if (btnId === 'add') { | ||
const actualZoom = map.getView().getZoom(); | ||
map.getView().setZoom(actualZoom + 1); | ||
} else if (btnId === 'minus') { | ||
const actualZoom = map.getView().getZoom(); | ||
map.getView().setZoom(actualZoom - 1); | ||
} else if (btnId === 'currentLocation') { | ||
setToggleCurrentLoc(!toggleCurrentLoc); | ||
dispatch(ProjectActions.ToggleGeolocationStatus(!geolocationStatus)); | ||
} else if (btnId === 'taskBoundries') { | ||
const layers = map.getAllLayers(); | ||
let extent; | ||
layers.map((layer) => { | ||
if (layer instanceof VectorLayer) { | ||
const layerName = layer.getProperties().name; | ||
if (layerName === 'project-area') { | ||
extent = layer.getSource().getExtent(); | ||
} | ||
} | ||
}); | ||
map.getView().fit(extent, { | ||
padding: [10, 10, 10, 10], | ||
}); | ||
} | ||
}; | ||
|
||
return ( | ||
<div className="fmtm-absolute fmtm-top-[20px] fmtm-right-5 fmtm-z-[99] fmtm-flex fmtm-flex-col fmtm-gap-4"> | ||
{btnList.map((btn) => ( | ||
<div key={btn.id}> | ||
<div | ||
className="fmtm-bg-white fmtm-rounded-full fmtm-p-2 hover:fmtm-bg-gray-100 fmtm-cursor-pointer fmtm-duration-300" | ||
onClick={() => handleOnClick(btn.id)} | ||
> | ||
{btn.icon} | ||
</div> | ||
</div> | ||
))} | ||
</div> | ||
); | ||
}; | ||
|
||
export default MapControlComponent; |
27 changes: 27 additions & 0 deletions
27
src/frontend/src/components/ProjectDetailsV2/MobileActivitiesContents.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
import React from 'react'; | ||
import ActivitiesPanel from './ActivitiesPanel'; | ||
import CoreModules from '../../shared/CoreModules'; | ||
|
||
const MobileActivitiesContents = ({ map, mainView, mapDivPostion }) => { | ||
const params = CoreModules.useParams(); | ||
const state = CoreModules.useAppSelector((state) => state.project); | ||
const defaultTheme = CoreModules.useAppSelector((state) => state.theme.hotTheme); | ||
|
||
return ( | ||
<div className="fmtm-w-full fmtm-bg-white fmtm-mb-[12vh]"> | ||
<CoreModules.Stack sx={{ display: 'flex', flexDirection: 'row', justifyContent: 'center' }}> | ||
<ActivitiesPanel | ||
params={params} | ||
state={state.projectTaskBoundries} | ||
defaultTheme={defaultTheme} | ||
map={map} | ||
view={mainView} | ||
mapDivPostion={mapDivPostion} | ||
states={state} | ||
/> | ||
</CoreModules.Stack> | ||
</div> | ||
); | ||
}; | ||
|
||
export default MobileActivitiesContents; |
Oops, something went wrong.