Compare commits

...

3 Commits

Author SHA1 Message Date
Pramod Mahajan
14cad23eec refactor: center chart using Bootstrap utilities and clean up layout 2025-07-07 16:27:46 +05:30
6f614334a8 modify content layout in project overview widget 2025-07-07 11:39:07 +05:30
b512a1dd3d Modify Project overview design
add project overview to dashboard
2025-07-07 10:04:13 +05:30
3 changed files with 192 additions and 67 deletions

View File

@ -9,6 +9,7 @@ import Teams from "./Teams";
import TasksCard from "./Tasks"; import TasksCard from "./Tasks";
import ProjectCompletionChart from "./ProjectCompletionChart"; import ProjectCompletionChart from "./ProjectCompletionChart";
import ProjectProgressChart from "./ProjectProgressChart"; import ProjectProgressChart from "./ProjectProgressChart";
import ProjectOverview from "../Project/ProjectOverview";
// import Attendance from "./Attendance"; // import Attendance from "./Attendance";
const Dashboard = () => { const Dashboard = () => {
@ -44,12 +45,12 @@ const Dashboard = () => {
<ProjectProgressChart /> <ProjectProgressChart />
</div> </div>
{/* <div className="col-xxl-6 col-lg-6"> <div className="col-xxl-6 col-lg-6">
<Attendance /> <ProjectOverview />
</div> */} </div>
</div> </div>
</div> </div>
); );
}; };
export default Dashboard; export default Dashboard;

View File

@ -71,8 +71,8 @@ const AboutProject = ({ data }) => {
</div> </div>
{data && ( {data && (
<div className="card mb-6"> <div className="card mb-6">
<div class="card-header text-start"> <div className="card-header text-start">
<h6 class="card-action-title mb-0"> <h6 className="card-action-title mb-0">
{" "} {" "}
<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>

View File

@ -1,20 +1,30 @@
import React from "react"; import React from "react";
import { useEffect, useState } from "react";
import { useSelector } from "react-redux";
import { import {
useEmployeesByProjectAllocated, useEmployeesByProjectAllocated,
useProjects, useProjects,
} from "../../hooks/useProjects"; } from "../../hooks/useProjects";
import ReactApexChart from "react-apexcharts";
import Chart from "react-apexcharts";
const ProjectOverview = ({ project }) => { const ProjectOverview = ({ project }) => {
const { projects } = useProjects(); const { projects } = useProjects();
const getProgress = (planned, completed) => { const [current_project, setCurrentProject] = useState(
return (completed * 100) / planned + "%"; projects.find((pro) => pro.id == project)
}; );
const getProgressInNumber = (planned, completed) => {
var number = (completed * 100) / planned; const selectedProject = useSelector(
return FormattedNumber(number); (store) => store.localVariables.projectId
);
const getProgressInPercentage = (planned, completed) => {
if (completed && planned) return (completed * 100) / planned;
else return 0;
}; };
const project_detail = projects.find((pro) => pro.id == project); //let project_detail = projects.find((pro) => pro.id == project);
// Utility function to check if a number has a decimal part // Utility function to check if a number has a decimal part
const hasDecimal = (num) => { const hasDecimal = (num) => {
@ -30,7 +40,7 @@ const ProjectOverview = ({ project }) => {
// Handle non-numeric values gracefully // Handle non-numeric values gracefully
if (isNaN(numericValue)) { if (isNaN(numericValue)) {
return <span>Invalid Number</span>; return 0;
} }
let options = {}; let options = {};
@ -58,6 +68,104 @@ const ProjectOverview = ({ project }) => {
return formattedString; return formattedString;
} }
const getRadialBarOptions = (percentage) => {
return {
chart: {
height: 350,
type: "radialBar",
sparkline: {
// Often used with gauges for a minimalist look
enabled: true,
},
},
plotOptions: {
radialBar: {
startAngle: -90, // Start the gauge from the left (bottom-left)
endAngle: 90, // End the gauge at the right (bottom-right)
hollow: {
size: "70%", // Size of the hollow part of the bar
},
dataLabels: {
show: true,
name: {
show: true,
fontSize: "16px",
fontFamily: "Inter, sans-serif",
color: "#6B7280", // Tailwind gray-500
offsetY: -10,
},
value: {
show: true,
fontSize: "28px",
fontFamily: "Inter, sans-serif",
color: "#374151", // Tailwind gray-700
offsetY: 20,
formatter: function (val) {
return FormattedNumber(val) + "%"; // Format value as percentage
},
},
},
track: {
background: "#E5E7EB", // Tailwind gray-200 for the track
strokeWidth: "97%",
margin: 5, // margin in between segments
dropShadow: {
enabled: true,
top: 2,
left: 0,
blur: 4,
opacity: 0.15,
},
},
},
},
fill: {
type: "gradient",
gradient: {
shade: "dark",
type: "horizontal",
shadeIntensity: 0.5,
gradientToColors: ["#6366F1"], // Tailwind indigo-500
inverseColors: true,
opacityFrom: 1,
opacityTo: 1,
stops: [0, 100],
},
},
stroke: {
lineCap: "round", // Rounded ends for the bar
},
labels: ["Progress"], // Label for the radial bar
series: [percentage], // The percentage value
};
};
const [radialPercentage, setRadialPercentage] = useState(75); // Initial percentage
const radialBarOptions = getRadialBarOptions(radialPercentage);
useEffect(() => {
if (current_project) {
let val = getProgressInPercentage(
current_project.plannedWork,
current_project.completedWork
);
setRadialPercentage(val);
} else setRadialPercentage(0);
}, [current_project]);
useEffect(() => {
setCurrentProject(projects.find((pro) => pro.id == selectedProject));
console.log(selectedProject);
if (current_project) {
let val = getProgressInPercentage(
current_project.plannedWork,
current_project.completedWork
);
setRadialPercentage(val);
} else setRadialPercentage(0);
}, [selectedProject]);
return ( return (
<div className="card mb-6"> <div className="card mb-6">
<div className="card-header text-start"> <div className="card-header text-start">
@ -67,60 +175,76 @@ const ProjectOverview = ({ project }) => {
<span className="ms-2">Project Statistics</span> <span className="ms-2">Project Statistics</span>
</h6> </h6>
</div> </div>
<div className="card-body"> <div className="card-body">
<ul className="list-unstyled mb-0 mt-3 pt-1"> <ul className="list-unstyled m-0 p-0">
<li className="d-flex align-items-center mb-3"> <li className="d-flex flex-wrap">
<i className="bx bx-check"></i> <div className="w-100 d-flex flex-wrap">
<span className="fw-medium mx-2">Task Planned:</span>{" "} {/* Centered Chart */}
<span>{project_detail?.plannedWork}</span> <div className="w-100 d-flex justify-content-center mb-3">
</li> <div >
<li className="d-flex align-items-center mb-3"> <Chart
<i className="bx bx-star"></i> options={radialBarOptions}
<span className="fw-medium mx-2">Task Completed:</span>{" "} series={radialBarOptions.series}
<span>{project_detail?.completedWork}</span> type="radialBar"
</li> height="100%"
<li className="d-flex align-items-center mb-3"> />
<i className="bx bx-user"></i> </div>
<span className="fw-medium mx-2">Current team Size:</span>{" "} </div>
<span>{project_detail?.teamSize}</span>
</li> {/* Info Section */}
<li className=" mb-3"> <div className="mb-2" style={{ flex: "1 1 auto" }}>
{project_detail && ( <div>
<> {/* Tasks Planned */}
<div className="d-flex text-end mb-2 mt-5"> <div className="d-flex align-items-center mb-3">
<small className="text-body text-muted "> <div className="avatar me-2">
{Math.floor( <span className="avatar-initial rounded-2 bg-label-primary">
getProgressInNumber( <i className="bx bx-check text-primary fs-4"></i>
project_detail.plannedWork, </span>
project_detail.completedWork </div>
) <div className="d-flex flex-column text-start">
) || 0}{" "} <small className="fw-bold">Tasks Planned</small>
% Completed <h5 className="mb-0">
</small> {FormattedNumber(current_project?.plannedWork)}
</div> </h5>
<div </div>
className="progress mb-4 rounded" </div>
style={{ height: "8px" }}
> {/* Tasks Completed */}
<div <div className="d-flex align-items-center mb-3">
className="progress-bar rounded" <div className="avatar me-2">
role="progressbar" <span className="avatar-initial rounded-2 bg-label-info">
style={{ <i className="bx bx-star text-info fs-4"></i>
width: getProgress( </span>
project_detail.plannedWork, </div>
project_detail.completedWork <div className="d-flex flex-column text-start">
), <small className="fw-bold">Tasks Completed</small>
}} <h5 className="mb-0">
aria-valuenow={project_detail.completedWork} {FormattedNumber(current_project?.completedWork)}
aria-valuemin="0" </h5>
aria-valuemax={project_detail.plannedWork} </div>
></div> </div>
</div>
</> {/* Team Size */}
)} <div className="d-flex align-items-center">
</li> <div className="avatar me-2">
</ul> <span className="avatar-initial rounded-2 bg-label-primary">
<i className="bx bx-group text-primary fs-4"></i>
</span>
</div>
<div className="d-flex flex-column text-start">
<small className="fw-bold">Current Team Size</small>
<h5 className="mb-0">
{FormattedNumber(current_project?.teamSize)}
</h5>
</div>
</div>
</div>
</div>
</div> </div>
</li>
</ul>
</div>
</div> </div>
); );
}; };