fixed directory header layout

This commit is contained in:
pramod.mahajan 2025-10-11 12:43:37 +05:30
parent 31882c3d12
commit 281a956ac8
6 changed files with 166 additions and 196 deletions

View File

@ -38,190 +38,136 @@ const usePagination = (data, itemsPerPage) => {
}; };
const AttendanceLog = ({ handleModalData, searchTerm, organizationId }) => { const AttendanceLog = ({ handleModalData, searchTerm, organizationId }) => {
// const selectedProject = useSelector( const selectedProject = useSelectedProject();
// (store) => store.localVariables.projectId const [dateRange, setDateRange] = useState({ startDate: "", endDate: "" });
// ); const dispatch = useDispatch();
const selectedProject = useSelectedProject(); const [loading, setLoading] = useState(false);
const [dateRange, setDateRange] = useState({ startDate: "", endDate: "" }); const [showPending, setShowPending] = useState(false);
const dispatch = useDispatch(); const [isRefreshing, setIsRefreshing] = useState(false);
const [loading, setLoading] = useState(false);
const [showPending, setShowPending] = useState(false);
const [isRefreshing, setIsRefreshing] = useState(false); const today = new Date();
const [processedData, setProcessedData] = useState([]); today.setHours(0, 0, 0, 0);
const today = new Date(); const yesterday = new Date();
today.setHours(0, 0, 0, 0); yesterday.setDate(yesterday.getDate() - 1);
const yesterday = new Date(); const isSameDay = (dateStr) => {
yesterday.setDate(yesterday.getDate() - 1); if (!dateStr) return false;
const d = new Date(dateStr);
d.setHours(0, 0, 0, 0);
return d.getTime() === today.getTime();
};
const isSameDay = (dateStr) => { const isBeforeToday = (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();
}; };
const isBeforeToday = (dateStr) => { const sortByName = (a, b) => {
if (!dateStr) return false; const nameA = (a.firstName + a.lastName).toLowerCase();
const d = new Date(dateStr); const nameB = (b.firstName + b.lastName).toLowerCase();
d.setHours(0, 0, 0, 0); return nameA.localeCompare(nameB);
return d.getTime() < today.getTime(); };
};
const sortByName = (a, b) => { const { data = [], isLoading, error, refetch, isFetching } = useAttendancesLogs(
const nameA = a.firstName.toLowerCase() + a.lastName.toLowerCase(); selectedProject,
const nameB = b.firstName.toLowerCase() + b.lastName.toLowerCase(); dateRange.startDate,
return nameA?.localeCompare(nameB); dateRange.endDate,
}; organizationId
);
const { const processedData = useMemo(() => {
data = [], const filteredData = showPending
isLoading, ? data.filter((item) => item.checkOutTime === null)
error, : data;
refetch,
isFetching,
} = useAttendancesLogs(
selectedProject,
dateRange.startDate,
dateRange.endDate,
organizationId
);
const filtering = (data) => {
const filteredData = showPending
? data.filter((item) => item.checkOutTime === null)
: data;
const group1 = filteredData const group1 = filteredData.filter((d) => d.activity === 1 && isSameDay(d.checkInTime)).sort(sortByName);
.filter((d) => d.activity === 1 && isSameDay(d.checkInTime)) const group2 = filteredData.filter((d) => d.activity === 4 && isSameDay(d.checkOutTime)).sort(sortByName);
.sort(sortByName); const group3 = filteredData.filter((d) => d.activity === 1 && isBeforeToday(d.checkInTime)).sort(sortByName);
const group2 = filteredData const group4 = filteredData.filter((d) => d.activity === 4 && isBeforeToday(d.checkOutTime));
.filter((d) => d.activity === 4 && isSameDay(d.checkOutTime)) const group5 = filteredData.filter((d) => d.activity === 2 && isBeforeToday(d.checkOutTime)).sort(sortByName);
.sort(sortByName); const group6 = filteredData.filter((d) => d.activity === 5).sort(sortByName);
const group3 = filteredData
.filter((d) => d.activity === 1 && isBeforeToday(d.checkInTime))
.sort(sortByName);
const group4 = filteredData.filter(
(d) => d.activity === 4 && isBeforeToday(d.checkOutTime)
);
const group5 = filteredData
.filter((d) => d.activity === 2 && isBeforeToday(d.checkOutTime))
.sort(sortByName);
const group6 = filteredData
.filter((d) => d.activity === 5)
.sort(sortByName);
const sortedList = [ const sortedList = [...group1, ...group2, ...group3, ...group4, ...group5, ...group6];
...group1,
...group2,
...group3,
...group4,
...group5,
...group6,
];
// Group by date const groupedByDate = sortedList.reduce((acc, item) => {
const groupedByDate = sortedList.reduce((acc, item) => { const date = (item.checkInTime || item.checkOutTime)?.split("T")[0];
const date = (item.checkInTime || item.checkOutTime)?.split("T")[0]; if (date) {
if (date) { acc[date] = acc[date] || [];
acc[date] = acc[date] || []; acc[date].push(item);
acc[date].push(item);
}
return acc;
}, {});
const sortedDates = Object.keys(groupedByDate).sort(
(a, b) => new Date(b) - new Date(a)
);
const finalData = sortedDates.flatMap((date) => groupedByDate[date]);
setProcessedData(finalData);
};
useEffect(() => {
filtering(data);
}, [data, showPending]);
// New useEffect to handle search filtering
const filteredSearchData = useMemo(() => {
if (!searchTerm) {
return processedData;
} }
const lowercasedSearchTerm = searchTerm.toLowerCase(); return acc;
return processedData.filter((item) => { }, {});
const fullName = `${item.firstName} ${item.lastName}`.toLowerCase();
return fullName.includes(lowercasedSearchTerm);
});
}, [processedData, searchTerm]);
const { const sortedDates = Object.keys(groupedByDate).sort((a, b) => new Date(b) - new Date(a));
currentPage, return sortedDates.flatMap((date) => groupedByDate[date]);
totalPages, }, [data, showPending]);
currentItems: paginatedAttendances,
paginate,
resetPage,
} = usePagination(filteredSearchData, 20);
useEffect(() => { const filteredSearchData = useMemo(() => {
resetPage(); if (!searchTerm) return processedData;
}, [filteredSearchData, resetPage]);
const handler = useCallback( const lowercased = searchTerm.toLowerCase();
(msg) => { return processedData.filter((item) =>
const { startDate, endDate } = dateRange; `${item.firstName} ${item.lastName}`.toLowerCase().includes(lowercased)
const checkIn = msg.response.checkInTime.substring(0, 10);
if (
selectedProject === msg.projectId &&
startDate <= checkIn &&
checkIn <= endDate
) {
queryClient.setQueriesData(["attendanceLogs"], (oldData) => {
if (!oldData) {
queryClient.invalidateQueries({ queryKey: ["attendanceLogs"] });
return;
}
const updatedAttendance = oldData.map((record) =>
record.id === msg.response.id
? { ...record, ...msg.response }
: record
);
filtering(updatedAttendance);
return updatedAttendance;
});
resetPage();
}
},
[selectedProject, dateRange, filtering, resetPage]
); );
}, [processedData, searchTerm]);
useEffect(() => { const {
eventBus.on("attendance_log", handler); currentPage,
return () => eventBus.off("attendance_log", handler); totalPages,
}, [handler]); currentItems: paginatedAttendances,
paginate,
resetPage,
} = usePagination(filteredSearchData, 20);
const employeeHandler = useCallback( useEffect(() => {
(msg) => { resetPage();
const { startDate, endDate } = dateRange; }, [filteredSearchData]);
if (data.some((item) => item.employeeId == msg.employeeId)) {
// dispatch( const handler = useCallback(
// fetchAttendanceData({ (msg) => {
// , const { startDate, endDate } = dateRange;
// fromDate: startDate, const checkIn = msg.response.checkInTime.substring(0, 10);
// toDate: endDate,
// }) if (selectedProject === msg.projectId && startDate <= checkIn && checkIn <= endDate) {
// ); queryClient.setQueriesData(["attendanceLogs"], (oldData) => {
if (!oldData) {
queryClient.invalidateQueries({ queryKey: ["attendanceLogs"] });
return;
}
return oldData.map((record) =>
record.id === msg.response.id ? { ...record, ...msg.response } : record
);
});
resetPage();
}
},
[selectedProject, dateRange, resetPage]
);
useEffect(() => {
eventBus.on("attendance_log", handler);
return () => eventBus.off("attendance_log", handler);
}, [handler]);
const employeeHandler = useCallback(
(msg) => {
const { startDate, endDate } = dateRange;
if (data.some((item) => item.employeeId == msg.employeeId)) {
refetch();
}
},
[data, refetch]
);
useEffect(() => {
eventBus.on("employee", employeeHandler);
return () => eventBus.off("employee", employeeHandler);
}, [employeeHandler]);
refetch();
}
},
[selectedProject, dateRange, data, refetch]
);
useEffect(() => {
eventBus.on("employee", employeeHandler);
return () => eventBus.off("employee", employeeHandler);
}, [employeeHandler]);
return ( return (
<> <>

View File

@ -33,7 +33,11 @@ const Regularization = ({
); );
useEffect(() => { useEffect(() => {
if(!regularizes) return
if(regularizes?.length) {
setregularizedList(regularizes); setregularizedList(regularizes);
}
}, [regularizes]); }, [regularizes]);
const sortByName = (a, b) => { const sortByName = (a, b) => {

View File

@ -70,12 +70,12 @@ const MasterModal = ({ modaldata, closeModal }) => {
}; };
return ( return (
<> <div className="p-2 p-md-1">
<div className="text-center"> <div className="text-center">
<p className="fs-5 fw-semibold" >{`${masterType, " ", modalType}`}</p> <p className="fs-5 fw-semibold" >{`${masterType, " ", modalType}`}</p>
</div> </div>
{ modalComponents[modalType] || null} { modalComponents[modalType] || null}
</> </div>
); );
}; };

View File

@ -162,29 +162,33 @@ export default function DirectoryPage({ IsPage = true, projectId = null }) {
</ul> </ul>
</div> </div>
<div className="mb-1 px-2 py-3"> <div className="mb-1 px-md-2 px-0 py-3">
<div className="d-flex align-items-center justify-content-between"> <div className="row">
<div className="d-flex align-items-center gap-3"> <div className="col-12 col-md-10 mb-2">
{activeTab === "notes" && ( {activeTab === "notes" && (
<input <div className="col-8 col-md-3">
<input
type="search" type="search"
className="form-control form-control-sm" className="form-control form-control-sm"
placeholder="Search notes..." placeholder="Search notes..."
value={searchNote} value={searchNote}
onChange={(e) => setSearchNote(e.target.value)} onChange={(e) => setSearchNote(e.target.value)}
/> />
</div>
)} )}
{activeTab === "contacts" && ( {activeTab === "contacts" && (
<div className="d-flex align-items-center gap-3"> <div className="d-flex align-items-center gap-3">
<div className="d-flex gap-2 align-items-center"> <div className="col-12 col-md-8 d-flex flex-row gap-2">
<input <div className="col-7 col-md-4">
<input
type="search" type="search"
className="form-control form-control-sm" className="form-control form-control-sm"
placeholder="Search contacts..." placeholder="Search contacts..."
value={searchContact} value={searchContact}
onChange={(e) => setsearchContact(e.target.value)} onChange={(e) => setsearchContact(e.target.value)}
/> />
</div>
<button <button
className={`btn btn-sm p-1 ${ className={`btn btn-sm p-1 ${
!gridView ? "btn-primary" : "btn-outline-primary" !gridView ? "btn-primary" : "btn-outline-primary"
@ -202,9 +206,7 @@ export default function DirectoryPage({ IsPage = true, projectId = null }) {
> >
<i className="bx bx-grid-alt"></i> <i className="bx bx-grid-alt"></i>
</button> </button>
</div> <div className="form-check form-switch d-flex align-items-center d-none d-md-flex">
<div className="form-check form-switch d-flex align-items-center">
<input <input
type="checkbox" type="checkbox"
className="form-check-input" className="form-check-input"
@ -220,11 +222,30 @@ export default function DirectoryPage({ IsPage = true, projectId = null }) {
{showActive ? "Active" : "Inactive"} Contacts {showActive ? "Active" : "Inactive"} Contacts
</label> </label>
</div> </div>
</div>
</div> </div>
)} )}
</div> </div>
<div className="col-12 col-md-2 d-flex justify-content-end align-items-center gap-2">
<div className="btn-group"> <div className={`form-check form-switch d-flex align-items-center ${activeTab === "contacts" ? " d-flex d-md-none m-0":"d-none" }`}>
<input
type="checkbox"
className="form-check-input"
role="switch"
id="inactiveEmployeesCheckbox"
checked={showActive}
onChange={(e) => setShowActive(e.target.checked)}
/>
<label
className="form-check-label ms-2"
htmlFor="inactiveEmployeesCheckbox"
>
{showActive ? "Active" : "Inactive"} Contacts
</label>
</div>
<div className=" btn-group">
<button <button
className="btn btn-sm btn-label-secondary dropdown-toggle" className="btn btn-sm btn-label-secondary dropdown-toggle"
type="button" type="button"
@ -244,6 +265,8 @@ export default function DirectoryPage({ IsPage = true, projectId = null }) {
</li> </li>
</ul> </ul>
</div> </div>
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@ -103,7 +103,7 @@ const MasterPage = () => {
if (menuErrorFlag || isMasterError) if (menuErrorFlag || isMasterError)
return ( return (
<div className="d-flex flex-column align-items-center justify-content-center py-5"> <div className="d-flex flex-column align-items-center justify-content-center py-1 py-md-5">
<h4 className="mb-3"> <h4 className="mb-3">
<i className="fa-solid fa-triangle-exclamation fs-5" /> Oops, an error <i className="fa-solid fa-triangle-exclamation fs-5" /> Oops, an error
occurred occurred
@ -161,21 +161,17 @@ const MasterPage = () => {
data={[{ label: "Home", link: "/dashboard" }, { label: "Masters" }]} data={[{ label: "Home", link: "/dashboard" }, { label: "Masters" }]}
/> />
<div className="row"> <div className="row page-min-h">
<div className="card"> <div className="card">
<div <div
className="card-datatable table-responsive py-10 mx-5 " className="card-datatable table-responsive py-2 py-md-10 mx-1 mx-md-5 "
style={{ overflow: "hidden" }} style={{ overflow: "hidden" }}
> >
<div className="row mb-2"> <div className="row mb-2">
<div className="col-md-3 col-sm-6"> <div className="col-12 col-md-3">
<select <select
className="form-select py-1 px-2" className="form-select py-1 px-2"
style={{
fontSize: "0.875rem",
height: "32px",
width: "190px",
}}
value={selectedMaster} value={selectedMaster}
onChange={(e) => dispatch(changeMaster(e.target.value))} onChange={(e) => dispatch(changeMaster(e.target.value))}
> >
@ -190,8 +186,8 @@ const MasterPage = () => {
)} )}
</select> </select>
</div> </div>
<div className="col-md-9 col-sm-6 d-flex justify-content-end align-items-center gap-2"> <div className="col-12 col-md-9 d-flex justify-content-end align-items-center gap-2 mt-2 mt-md-0">
<div className="w-25"> <div className="col-6 col-md-3">
<input <input
type="search" type="search"
className="form-control form-control-sm" className="form-control form-control-sm"
@ -207,8 +203,8 @@ const MasterPage = () => {
handleModalData(selectedMaster, null, selectedMaster) handleModalData(selectedMaster, null, selectedMaster)
} }
> >
<i className="bx bx-plus-circle me-2"></i>Add{" "} <i className="bx bx-plus-circle me-2"></i> <span className="d-none d-md-inline-block">Add{" "}
{selectedMaster} {selectedMaster}</span>
</button> </button>
)} )}
</div> </div>

View File

@ -92,7 +92,8 @@ const MasterTable = ({ data, columns, loading, handleModalData }) => {
{" "} {" "}
{selectedMaster === "Activity" ? "Activity" : "Name"} {selectedMaster === "Activity" ? "Activity" : "Name"}
</th> </th>
<th className="text-start"> <th className="text-start d-none d-md-table-cell">
{" "} {" "}
{selectedMaster === "Activity" {selectedMaster === "Activity"
? "Unit" ? "Unit"
@ -108,12 +109,12 @@ const MasterTable = ({ data, columns, loading, handleModalData }) => {
<tbody> <tbody>
{currentItems.length > 0 ? ( {currentItems.length > 0 ? (
currentItems.map((item, index) => ( currentItems.map((item, index) => (
<tr key={index}> <tr key={index} >
<td style={{ width: "20px" }}> <td style={{ width: "20px" }} className="py-3">
<i className="bx bx-right-arrow-alt"></i> <i className="bx bx-right-arrow-alt"></i>
</td> </td>
{updatedColumns.map((col) => ( {updatedColumns.map((col) => (
<td className="text-start mx-2" key={col.key}> <td className={`text-start mx-2 py-3 ${col.key === "description" && "d-none d-md-table-cell"}`} key={col.key} >
{col.key === "description" ? ( {col.key === "description" ? (
item[col.key] !== undefined && item[col.key] !== undefined &&
item[col.key] !== null ? ( item[col.key] !== null ? (
@ -133,7 +134,7 @@ const MasterTable = ({ data, columns, loading, handleModalData }) => {
)} )}
</td> </td>
))} ))}
<td className={!hasMasterPermission ? "d-none" : ""}> <td className={!hasMasterPermission ? "d-none" : "py-3"}>
{(selectedMaster === "Application Role" || {(selectedMaster === "Application Role" ||
selectedMaster === "Work Category") && selectedMaster === "Work Category") &&
item?.isSystem ? ( item?.isSystem ? (