149 lines
5.4 KiB
TypeScript
149 lines
5.4 KiB
TypeScript
import { ReactNode, useEffect, useState } from "react";
|
|
import MainBtn from "../MainBtn";
|
|
|
|
const data1:{ name: string; email: string; status: string; location: string; }[] = [];
|
|
|
|
type PaginatedListProps = {
|
|
data: any,
|
|
itemsPerPage: number,
|
|
filterItem?: string[],
|
|
titleClass?:string,
|
|
children: (data:any) => ReactNode;
|
|
}
|
|
|
|
export default function TableWrapper({
|
|
data = data1,
|
|
itemsPerPage = 5,
|
|
filterItem,
|
|
children,
|
|
}:PaginatedListProps) {
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
const [searchTerm, setSearchTerm] = useState("");
|
|
const [filteredData, setFilteredData] = useState(data);
|
|
|
|
const [currentPage, setCurrentPage] = useState(0);
|
|
const [newData, setNewData] = useState<{ name: string; email: string; status: string; location: string; }[]>([]);
|
|
|
|
const numberOfSelection = itemsPerPage;
|
|
|
|
const handlePrev = () => {
|
|
if (currentPage != 0) {
|
|
setCurrentPage((prev:any) => prev - numberOfSelection);
|
|
}
|
|
};
|
|
const handleNext = () => {
|
|
if (currentPage < data.length) {
|
|
setCurrentPage((prev:any) => prev + numberOfSelection);
|
|
}
|
|
};
|
|
|
|
const handleSearch = ({ target: { value } }:{target: {value:string}}, name:string) => {
|
|
setSearchTerm(value);
|
|
let newFilteredData:any = data.filter((item:any) =>
|
|
item[name].toLowerCase().startsWith(value.toLowerCase())
|
|
);
|
|
setFilteredData(newFilteredData);
|
|
setCurrentPage(0);
|
|
};
|
|
|
|
useEffect(() => {
|
|
setIsLoading(true)
|
|
setTimeout(()=>{
|
|
setNewData(
|
|
filteredData?.slice(currentPage, numberOfSelection + currentPage)
|
|
);
|
|
setIsLoading(false)
|
|
},1000)
|
|
}, [currentPage, filteredData]);
|
|
|
|
useEffect(()=>{
|
|
setCurrentPage(0)
|
|
},[itemsPerPage])
|
|
|
|
return (
|
|
<div className="p-2 w-full bg-white border-b dark:bg-gray-800 rounded-md">
|
|
{data.length > 0 && filterItem && (
|
|
<div className="mb-10 flex justify-end items-center gap-2">
|
|
{filterItem.map((item, index) => (
|
|
<label
|
|
key={index}
|
|
className="flex flex-col sm:flex-row items-center gap-2 text-slate-600 dark:text-slate-100 transition-all duration-500"
|
|
>
|
|
Search by {item[0].toUpperCase() + item.slice(1)}
|
|
<input
|
|
name={item}
|
|
type="text"
|
|
className="py-1 px-2 text-sm min-w-[100px] text-black dark:text-white bg-white dark:bg-slate-800 rounded-full border-0 outline-none ring-1 ring-slate-300 dark:ring-white transition-all duration-500"
|
|
value={searchTerm}
|
|
onChange={(e) => {
|
|
handleSearch(e, item);
|
|
}}
|
|
/>
|
|
</label>
|
|
))}
|
|
</div>
|
|
)}
|
|
|
|
<div className="flex flex-col">
|
|
<div className="w-full">
|
|
{children({ data: newData })}
|
|
</div>
|
|
|
|
{/* PAGINATION BUTTON */}
|
|
{(newData.length > 0 && data.length > itemsPerPage) &&
|
|
<div className='p-2 w-full flex flex-col lg:flex-row justify-center items-center gap-3 md:gap-6'>
|
|
<div className="text-sm text-center lg:text-left font-normal text-gray-500 dark:text-gray-400 block w-full">Showing <span className="font-semibold text-gray-900 dark:text-white">
|
|
{isLoading ? '----' : `${currentPage + 1}-${currentPage + numberOfSelection >= data.length ? data.length : (currentPage + numberOfSelection)}`}</span> of <span className="font-semibold text-gray-900 dark:text-white">{data.length}</span>
|
|
</div>
|
|
<div className='flex items-center gap-3 md:gap-6'>
|
|
<MainBtn
|
|
onClick={handlePrev}
|
|
text='Prev'
|
|
className={`${currentPage == 0 ? 'bg-sky-600/50 pointer-events-none' : 'bg-sky-600'} text-white-light`}
|
|
disabled={isLoading}
|
|
/>
|
|
<MainBtn
|
|
onClick={handleNext}
|
|
text='Next'
|
|
className={`${currentPage + numberOfSelection >= data.length ? 'bg-sky-600/50 pointer-events-none' : 'bg-sky-600'} text-white-light`}
|
|
disabled={isLoading}
|
|
/>
|
|
</div>
|
|
</div>
|
|
}
|
|
</div>
|
|
|
|
{/* show prev and next button if data exist */}
|
|
{/* {data.length > 0 && (
|
|
{data.length && data.map((item, index)=>{
|
|
item = item
|
|
if(index%itemsPerPage == 0 && index >= currentPage && index <= currentPage+itemsPerPage){
|
|
return (
|
|
<button
|
|
key={index}
|
|
onClick={handleNext}
|
|
className={`w-6 h-6 md:w-12 md:h-12 text-sm md:text-lg rounded-full flex justify-center items-center border transition-all duration-300 ${
|
|
currentPage != index
|
|
? "text-slate-400 border-slate-400 dark:text-slate-400 dark:border-slate-400"
|
|
: "text-slate-600 border-slate-600 dark:text-white dark:border-white pointer-events-none"
|
|
}`}
|
|
>
|
|
{index/itemsPerPage +1}
|
|
</button>
|
|
)
|
|
}
|
|
})}
|
|
</div>
|
|
)} */}
|
|
{isLoading && <TableIsLoading />}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const TableIsLoading = () => {
|
|
return (
|
|
<div className="w-full fixed z-[991] inset-0 flex justify-center items-center bg-white/50">
|
|
<p className="rounded-md shadow-md p-4 bg-white/90 dark:bg-gray-900 text-brown dark:text-white">Loading...</p>
|
|
</div>
|
|
)
|
|
} |