Creating a new weidget Attendance Overview and accroding to project selection data will seen.

This commit is contained in:
Kartik sharma 2025-07-16 16:14:52 +05:30 committed by pramod.mahajan
parent b502b6b31e
commit c1fb50c667
11 changed files with 408 additions and 132 deletions

View File

@ -836,7 +836,7 @@ progress {
} }
.row { .row {
--bs-gutter-x: 0.500rem; --bs-gutter-x: 1.500rem;
--bs-gutter-y: 0; --bs-gutter-y: 0;
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;

View File

@ -0,0 +1,17 @@
import React from "react";
const ChartSkeleton = () => {
return (
<div className="w-100">
<div className="placeholder-glow mb-3" style={{ width: "100%" }}>
<span className="placeholder col-6" style={{ height: "50px", display: "block" }}></span>
</div>
<div
className="bg-secondary bg-opacity-10 rounded"
style={{ height: "300px", width: "100%" }}
/>
</div>
);
};
export default ChartSkeleton;

View File

@ -0,0 +1,14 @@
const flatColors = [
"#E57373", "#64B5F6", "#81C784", "#FFB74D",
"#FF8A65", "#4DB6AC", "#A1887F", "#DCE775",
"#7986CB", "#AED581", "#4FC3F7", "#F06292", "#E0E0E0",
"#FFF176", "#A5D6A7", "#90CAF9", "#FFAB91",
"#E6EE9C", "#FFCC80", "#80DEEA", "#B0BEC5",
"#EF9A9A", "#FFCDD2", "#C5CAE9", "#F8BBD0", "#D1C4E9",
"#FFF9C4", "#C8E6C9", "#BBDEFB", "#FFECB3",
"#B2EBF2", "#CFD8DC", "#FBE9E7", "#FFFDE7",
"#DCEDC8", "#B3E5FC", "#FFF3E0", "#FCE4EC",
"#E0F7FA", "#ECEFF1", "#FFE0B2", "#FFD54F", "#FFA726",
];
export default flatColors;

View File

@ -0,0 +1,163 @@
import React, { useState, useMemo } from "react";
import { useSelector } from "react-redux";
import ReactApexChart from "react-apexcharts";
import { useAttendanceOverviewData } from "../../hooks/useDashboard_Data";
import flatColors from "../Charts/flatColor";
import ChartSkeleton from "../Charts/Skelton";
const formatDate = (dateStr) => {
const date = new Date(dateStr);
return date.toLocaleDateString("en-GB", {
day: "2-digit",
month: "long",
});
};
const AttendanceOverview = () => {
const [dayRange, setDayRange] = useState(7);
const [view, setView] = useState("chart");
const projectId = useSelector((store) => store.localVariables.projectId);
const { attendanceOverviewData, loading, error } = useAttendanceOverviewData(projectId, dayRange);
const { tableData, roles, dates } = useMemo(() => {
const map = new Map();
attendanceOverviewData.forEach((entry) => {
const date = formatDate(entry.date);
if (!map.has(date)) map.set(date, {});
map.get(date)[entry.role.trim()] = entry.present;
});
const uniqueRoles = [...new Set(attendanceOverviewData.map((e) => e.role.trim()))];
const sortedDates = [...map.keys()];
const data = sortedDates.map((date) => {
const row = { date };
uniqueRoles.forEach((role) => {
row[role] = map.get(date)?.[role] ?? 0;
});
return row;
});
return {
tableData: data,
roles: uniqueRoles,
dates: sortedDates,
};
}, [attendanceOverviewData]);
const chartSeries = roles.map((role) => ({
name: role,
data: tableData.map((row) => row[role]),
}));
const chartOptions = {
chart: {
type: "bar",
stacked: true,
height: 400,
toolbar: { show: false },
},
plotOptions: {
bar: {
borderRadius: 8,
columnWidth: "60%",
},
},
xaxis: {
categories: tableData.map((row) => row.date),
},
legend: {
position: "bottom",
},
fill: {
opacity: 1,
},
colors: roles.map((_, i) => flatColors[i % flatColors.length]),
};
return (
<div
className="bg-white p-4 mt-5 rounded shadow d-flex flex-column"
>
{/* Header */}
<div className="d-flex justify-content-between align-items-center mb-3">
<div className="card-title mb-0 text-start">
<h5 className="mb-1">Attendance Overview</h5>
<p className="card-subtitle">Role-wise present count</p>
</div>
<div className="d-flex gap-2">
<select
className="form-select form-select-sm"
value={dayRange}
onChange={(e) => setDayRange(Number(e.target.value))}
>
<option value={7}>Last 7 Days</option>
<option value={15}>Last 15 Days</option>
<option value={30}>Last 30 Days</option>
</select>
<button
className={`btn btn-sm ${view === "chart" ? "btn-primary" : "btn-outline-primary"}`}
onClick={() => setView("chart")}
>
📊
</button>
<button
className={`btn btn-sm ${view === "table" ? "btn-primary" : "btn-outline-primary"}`}
onClick={() => setView("table")}
>
📋
</button>
</div>
</div>
{/* Content */}
<div className="flex-grow-1 d-flex align-items-center justify-content-center">
{loading ? (
<ChartSkeleton />
) : error ? (
<p className="text-danger">{error}</p>
) : view === "chart" ? (
<div className="w-100">
<ReactApexChart
options={chartOptions}
series={chartSeries}
type="bar"
height={400}
/>
</div>
) : (
<div
className="table-responsive w-100"
style={{ maxHeight: "350px", overflowY: "auto" }}
>
<table className="table table-bordered table-sm text-center align-middle mb-0">
<thead className="table-light" style={{ position: "sticky", top: 0, zIndex: 1 }}>
<tr>
<th style={{ background: "#f8f9fa" }}>Role</th>
{dates.map((date, idx) => (
<th key={idx} style={{ background: "#f8f9fa" }}>{date}</th>
))}
</tr>
</thead>
<tbody>
{roles.map((role) => (
<tr key={role}>
<td>{role}</td>
{tableData.map((row, idx) => (
<td key={idx}>{row[role]}</td>
))}
</tr>
))}
</tbody>
</table>
</div>
)}
</div>
</div>
);
};
export default AttendanceOverview;

View File

@ -1,71 +1,69 @@
import React from "react"; import React from "react";
import { useSelector } from "react-redux"; // Import useSelector to access Redux state import { useSelector } from "react-redux";
import { import {
useDashboardProjectsCardData, useDashboardProjectsCardData,
useDashboardTeamsCardData, useDashboardTeamsCardData,
useDashboardTasksCardData, useDashboardTasksCardData,
useAttendanceOverviewData
} from "../../hooks/useDashboard_Data"; } from "../../hooks/useDashboard_Data";
import Projects from "./Projects"; import Projects from "./Projects";
import Teams from "./Teams"; 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 ProjectOverview from "../Project/ProjectOverview";
// import Attendance from "./Attendance"; import AttendanceOverview from "./AttendanceChart";
const Dashboard = () => { const Dashboard = () => {
const { projectsCardData } = useDashboardProjectsCardData(); const { projectsCardData } = useDashboardProjectsCardData();
const { teamsCardData } = useDashboardTeamsCardData(); const { teamsCardData } = useDashboardTeamsCardData();
const { tasksCardData } = useDashboardTasksCardData(); const { tasksCardData } = useDashboardTasksCardData();
// Get the selected project ID from Redux store // Get the selected project ID from Redux store
const selectedProjectId = useSelector( const projectId = useSelector((store) => store.localVariables.projectId);
(store) => store.localVariables.projectId const isAllProjectsSelected = projectId === null;
);
// Determine if "All Projects" is selected return (
// selectedProjectId will be null when "All Projects" is chosen <div className="container-fluid mt-5">
const isAllProjectsSelected = selectedProjectId === null; <div className="row gy-4">
{isAllProjectsSelected && (
<div className="col-sm-6 col-lg-4">
<Projects projectsCardData={projectsCardData} />
</div>
)}
return ( <div className={`${!isAllProjectsSelected ? "col-sm-6 col-lg-6" : "col-sm-6 col-lg-4"}`}>
<div className="container-fluid mt-3"> <Teams teamsCardData={teamsCardData} />
<div className="row gy-4"> </div>
{isAllProjectsSelected && ( <div className={`${!isAllProjectsSelected ? "col-sm-6 col-lg-6" : "col-sm-6 col-lg-4"}`}>
<div className="col-sm-6 col-lg-4"> <TasksCard tasksCardData={tasksCardData} />
<Projects projectsCardData={projectsCardData} /> </div>
</div>
)}
{isAllProjectsSelected && (
<div className="col-xxl-6 col-lg-6">
<ProjectCompletionChart />
</div>
)}
<div className={`${!isAllProjectsSelected ? "col-sm-6 col-lg-6":"col-sm-6 col-lg-4"}`}> {!isAllProjectsSelected && (
<Teams teamsCardData={teamsCardData} /> <div className="col-xxl-6 col-lg-6">
</div> <ProjectOverview />
</div>
)}
<div className={`${!isAllProjectsSelected ? "col-sm-6 col-lg-6":"col-sm-6 col-lg-4"}`}> <div className="col-xxl-6 col-lg-6">
<TasksCard tasksCardData={tasksCardData} /> <ProjectProgressChart />
</div> </div>
{!isAllProjectsSelected && (
<div className="col-xxl-6 col-lg-6">
{isAllProjectsSelected && ( <AttendanceOverview /> {/* ✅ Removed unnecessary projectId prop */}
<div className="col-xxl-6 col-lg-6"> </div>
<ProjectCompletionChart /> )}
</div> </div>
)} </div>
);
{! isAllProjectsSelected && (
<div className="col-xxl-6 col-lg-6">
<ProjectOverview />
</div>
)}
<div className="col-xxl-6 col-lg-6">
<ProjectProgressChart />
</div>
</div>
</div>
);
}; };
export default Dashboard; export default Dashboard;

View File

@ -1,8 +1,11 @@
import React from "react"; import React from "react";
import { useSelector } from "react-redux";
import { useDashboardTasksCardData } from "../../hooks/useDashboard_Data"; import { useDashboardTasksCardData } from "../../hooks/useDashboard_Data";
const TasksCard = () => { const TasksCard = () => {
const { tasksCardData } = useDashboardTasksCardData(); const projectId = useSelector((store) => store.localVariables?.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">
@ -11,20 +14,34 @@ const TasksCard = () => {
<i className="bx bx-task text-success"></i> Tasks <i className="bx bx-task text-success"></i> Tasks
</h5> </h5>
</div> </div>
<div className="d-flex justify-content-around align-items-start mt-n2">
<div> {loading ? (
<h4 className="mb-0 fw-bold"> // Loader will be displayed when loading is true
{tasksCardData.totalTasks?.toLocaleString()} <div className="d-flex justify-content-center align-items-center flex-grow-1">
</h4> <div className="spinner-border text-primary" role="status">
<small className="text-muted">Total</small> <span className="visually-hidden">Loading...</span>
</div>
</div> </div>
<div> ) : error ? (
<h4 className="mb-0 fw-bold"> // Error message if there's an error
{tasksCardData.completedTasks?.toLocaleString()} <div className="text-danger flex-grow-1 d-flex justify-content-center align-items-center">{error}</div>
</h4> ) : (
<small className="text-muted">Completed</small> // Actual data when loaded successfully
<div className="d-flex justify-content-around align-items-start mt-n2">
<div>
<h4 className="mb-0 fw-bold">
{tasksCardData?.totalTasks?.toLocaleString()}
</h4>
<small className="text-muted">Total</small>
</div>
<div>
<h4 className="mb-0 fw-bold">
{tasksCardData?.completedTasks?.toLocaleString()}
</h4>
<small className="text-muted">Completed</small>
</div>
</div> </div>
</div> )}
</div> </div>
); );
}; };

View File

@ -1,29 +1,33 @@
import React, { useCallback, useEffect, useState } from "react"; import React, { useCallback, useEffect, useState } from "react";
import { useSelector } from "react-redux";
import { useDashboardTeamsCardData } from "../../hooks/useDashboard_Data"; import { useDashboardTeamsCardData } from "../../hooks/useDashboard_Data";
import eventBus from "../../services/eventBus"; import eventBus from "../../services/eventBus";
const Teams = () => { const Teams = () => {
const { teamsCardData } = useDashboardTeamsCardData(); const projectId = useSelector((store) => store.localVariables?.projectId);
const[totalEmployees,setTotalEmployee] = useState(0); const { teamsCardData, loading, error } = useDashboardTeamsCardData(projectId);
const[inToday,setInToday] = useState(0);
useEffect(() =>{ const [totalEmployees, setTotalEmployee] = useState(0);
setTotalEmployee(teamsCardData.totalEmployees) const [inToday, setInToday] = useState(0);
setInToday(teamsCardData.inToday)
},[teamsCardData.totalEmployees,teamsCardData.inToday]) // Update state when API data arrives
useEffect(() => {
setTotalEmployee(teamsCardData?.totalEmployees || 0);
setInToday(teamsCardData?.inToday || 0);
}, [teamsCardData]);
// Handle real-time updates via eventBus
const handler = useCallback((msg) => {
if (msg.activity === 1) {
setInToday((prev) => prev + 1);
}
}, []);
const handler = useCallback(
(msg) => {
if (msg.activity == 1) {
setInToday(prev => prev + 1);
}
},
[inToday]
);
useEffect(() => { useEffect(() => {
eventBus.on("attendance", handler); eventBus.on("attendance", handler);
return () => eventBus.off("attendance", handler); return () => eventBus.off("attendance", handler);
}, [handler]); }, [handler]);
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">
<div className="d-flex justify-content-start align-items-center mb-3"> <div className="d-flex justify-content-start align-items-center mb-3">
@ -31,20 +35,30 @@ const Teams = () => {
<i className="bx bx-group text-warning"></i> Teams <i className="bx bx-group text-warning"></i> Teams
</h5> </h5>
</div> </div>
<div className="d-flex justify-content-around align-items-start mt-n2">
<div> {loading ? (
<h4 className="mb-0 fw-bold"> // Blue spinner loader
{totalEmployees?.toLocaleString()} <div className="d-flex justify-content-center align-items-center flex-grow-1">
</h4> <div className="spinner-border text-primary" role="status">
<small className="text-muted">Total Employees</small> <span className="visually-hidden">Loading...</span>
</div>
</div> </div>
<div> ) : error ? (
<h4 className="mb-0 fw-bold"> // Error message if data fetching fails
{inToday?.toLocaleString()} <div className="text-danger flex-grow-1 d-flex justify-content-center align-items-center">{error}</div>
</h4> ) : (
<small className="text-muted">In Today</small> // Display data once loaded
<div className="d-flex justify-content-around align-items-start mt-n2">
<div>
<h4 className="mb-0 fw-bold">{totalEmployees.toLocaleString()}</h4>
<small className="text-muted">Total Employees</small>
</div>
<div>
<h4 className="mb-0 fw-bold">{inToday.toLocaleString()}</h4>
<small className="text-muted">In Today</small>
</div>
</div> </div>
</div> )}
</div> </div>
); );
}; };

View File

@ -165,7 +165,7 @@ const ProjectOverview = ({ project }) => {
}, [selectedProject]); }, [selectedProject]);
return ( return (
<div className="card mb-6"> <div className="card mb-6" style={{ minHeight: "490px" }}>
<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">
{" "} {" "}

View File

@ -39,32 +39,32 @@ export const useDashboard_Data = ({ days, FromDate, projectId }) => {
export const useDashboard_AttendanceData = (date, projectId) => { export const useDashboard_AttendanceData = (date, projectId) => {
const [dashboard_Attendancedata, setDashboard_AttendanceData] = useState([]); const [dashboard_Attendancedata, setDashboard_AttendanceData] = useState([]);
const [isLineChartLoading, setLoading] = useState(false); const [isLineChartLoading, setLoading] = useState(false);
const [error, setError] = useState(""); const [error, setError] = useState("");
useEffect(() => { useEffect(() => {
const fetchData = async () => { const fetchData = async () => {
setLoading(true); setLoading(true);
setError(""); setError("");
try { try {
const response = await GlobalRepository.getDashboardAttendanceData(date,projectId); // date in 2nd param const response = await GlobalRepository.getDashboardAttendanceData(date, projectId); // date in 2nd param
setDashboard_AttendanceData(response.data); setDashboard_AttendanceData(response.data);
} catch (err) { } catch (err) {
setError("Failed to fetch dashboard data."); setError("Failed to fetch dashboard data.");
console.error(err); console.error(err);
} finally { } finally {
setLoading(false); setLoading(false);
} }
}; };
if (date && projectId !== null) { if (date && projectId !== null) {
fetchData(); fetchData();
} }
}, [date, projectId]); }, [date, projectId]);
return { dashboard_Attendancedata, isLineChartLoading: isLineChartLoading, error }; return { dashboard_Attendancedata, isLineChartLoading: isLineChartLoading, error };
}; };
@ -97,36 +97,38 @@ export const useDashboardProjectsCardData = () => {
}; };
// 🔹 Dashboard Teams Card Data Hook // 🔹 Dashboard Teams Card Data Hook
export const useDashboardTeamsCardData = () => { export const useDashboardTeamsCardData = (projectId) => {
const [teamsCardData, setTeamsData] = useState([]); const [teamsCardData, setTeamsData] = useState({});
const [loading, setLoading] = useState(false); const [loading, setLoading] = useState(false);
const [error, setError] = useState(""); const [error, setError] = useState("");
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(); const response = await GlobalRepository.getDashboardTeamsCardData(projectId);
setTeamsData(response.data); setTeamsData(response.data); // Handle undefined/null
} catch (err) { } catch (err) {
setError("Failed to fetch teams card data."); setError("Failed to fetch teams card data.");
console.error(err); console.error(err);
setTeamsData({});
} finally { } finally {
setLoading(false); setLoading(false);
} }
}; };
fetchTeamsData(); fetchTeamsData();
}, []); }, [projectId]);
return { teamsCardData, loading, error }; return { teamsCardData, loading, error };
}; };
// 🔹 Dashboard Tasks Card Data Hook export const useDashboardTasksCardData = (projectId) => {
export const useDashboardTasksCardData = () => { const [tasksCardData, setTasksData] = useState({});
const [tasksCardData, setTasksData] = useState([]);
const [loading, setLoading] = useState(false); const [loading, setLoading] = useState(false);
const [error, setError] = useState(""); const [error, setError] = useState("");
@ -136,18 +138,47 @@ export const useDashboardTasksCardData = () => {
setError(""); setError("");
try { try {
const response = await GlobalRepository.getDashboardTasksCardData(); const response = await GlobalRepository.getDashboardTasksCardData(projectId);
setTasksData(response.data); setTasksData(response.data);
} catch (err) { } catch (err) {
setError("Failed to fetch tasks card data."); setError("Failed to fetch tasks card data.");
console.error(err); console.error(err);
setTasksData({});
} finally { } finally {
setLoading(false); setLoading(false);
} }
}; };
fetchTasksData(); fetchTasksData();
}, []); }, [projectId]);
return { tasksCardData, loading, error }; return { tasksCardData, loading, error };
}; };
export const useAttendanceOverviewData = (projectId, days) => {
const [attendanceOverviewData, setAttendanceOverviewData] = useState([]);
const [loading, setLoading] = useState(false);
const [error, setError] = useState("");
useEffect(() => {
if (!projectId || !days) return;
const fetchAttendanceOverview = async () => {
setLoading(true);
setError("");
try {
const response = await GlobalRepository.getAttendanceOverview(projectId, days);
setAttendanceOverviewData(response.data);
} catch (err) {
setError("Failed to fetch attendance overview data.");
} finally {
setLoading(false);
}
};
fetchAttendanceOverview();
}, [projectId, days]);
return { attendanceOverviewData, loading, error };
};

View File

@ -1,4 +1,4 @@
import { useSelector } from "react-redux"; // Import useSelector import { useSelector, useDispatch } from "react-redux"; // Import useSelector
import React, { useState, useEffect, useCallback } from "react"; import React, { useState, useEffect, useCallback } from "react";
import ProjectOverview from "../../components/Project/ProjectOverview"; import ProjectOverview from "../../components/Project/ProjectOverview";
@ -22,14 +22,25 @@ import { ComingSoonPage } from "../Misc/ComingSoonPage";
import Directory from "../Directory/Directory"; import Directory from "../Directory/Directory";
import eventBus from "../../services/eventBus"; import eventBus from "../../services/eventBus";
import ProjectProgressChart from "../../components/Dashboard/ProjectProgressChart"; import ProjectProgressChart from "../../components/Dashboard/ProjectProgressChart";
import { useProjectName } from "../../hooks/useProjects";
import AttendanceOverview from "../../components/Dashboard/AttendanceChart";
const ProjectDetails = () => { const ProjectDetails = () => {
const projectId = useSelector((store) => store.localVariables.projectId); const projectId = useSelector((store) => store.localVariables.projectId);
const { projectNames, fetchData } = useProjectName();
const dispatch = useDispatch()
useEffect(() => {
if (projectId == null) {
dispatch(setProjectId(projectNames[0]?.id));
}
}, [projectNames])
const { const {
projects_Details, projects_Details,
loading: projectLoading, loading: projectLoading,
error: projectError, error: projectError,
refetch, refetch,
@ -71,6 +82,7 @@ const ProjectDetails = () => {
</div> </div>
<div className="col-lg-8 col-md-7 mt-5"> <div className="col-lg-8 col-md-7 mt-5">
<ProjectProgressChart ShowAllProject="false" DefaultRange="1M" /> <ProjectProgressChart ShowAllProject="false" DefaultRange="1M" />
<AttendanceOverview />
</div> </div>
</div> </div>
</> </>

View File

@ -26,12 +26,22 @@ const GlobalRepository = {
getDashboardProjectsCardData: () => { getDashboardProjectsCardData: () => {
return api.get(`/api/Dashboard/projects`); return api.get(`/api/Dashboard/projects`);
}, },
getDashboardTeamsCardData: () => {
return api.get(`/api/Dashboard/teams`); getDashboardTeamsCardData: (projectId) => {
}, const url = projectId
getDashboardTasksCardData: () => { ? `/api/Dashboard/teams?projectId=${projectId}`
return api.get(`/api/Dashboard/tasks`); : `/api/Dashboard/teams`;
}, return api.get(url);
},
getDashboardTasksCardData: (projectId) => {
const url = projectId
? `/api/Dashboard/tasks?projectId=${projectId}`
: `/api/Dashboard/tasks`;
return api.get(url);
},
getAttendanceOverview:(projectId,days)=>api.get(`/api/dashboard/attendance-overview/${projectId}?days=${days}`)
}; };