Spaces:
Paused
Paused
File size: 2,204 Bytes
1c72248 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 |
import Loading from './Loading';
import classNames from 'classnames';
export interface TableColumn {
title: string;
key: string;
render?: (row: any) => React.ReactNode;
className?: string;
}
interface TableRow {
[key: string]: any;
}
interface TableProps {
columns: TableColumn[];
rows: TableRow[];
isLoading: boolean;
onRefresh: () => void;
}
export default function UniversalTable({ columns, rows, isLoading, onRefresh = () => {} }: TableProps) {
return (
<div className="w-full bg-gray-900 rounded-md shadow-md">
{isLoading ? (
<div className="p-4 flex justify-center">
<Loading />
</div>
) : rows.length === 0 ? (
<div className="p-6 text-center text-gray-400">
<p className="text-sm">Empty</p>
<button
onClick={() => onRefresh()}
className="mt-2 px-3 py-1 text-xs bg-gray-800 hover:bg-gray-700 text-gray-300 rounded transition-colors"
>
Refresh
</button>
</div>
) : (
<div className="overflow-x-auto">
<table className="w-full text-sm text-left text-gray-300">
<thead className="text-xs uppercase bg-gray-800 text-gray-400">
<tr>
{columns.map(column => (
<th key={column.key} className="px-3 py-2">
{column.title}
</th>
))}
</tr>
</thead>
<tbody>
{rows?.map((row, index) => {
// Style for alternating rows
const rowClass = index % 2 === 0 ? 'bg-gray-900' : 'bg-gray-800';
return (
<tr key={index} className={`${rowClass} border-b border-gray-700 hover:bg-gray-700`}>
{columns.map(column => (
<td key={column.key} className={classNames('px-3 py-2', column.className)}>
{column.render ? column.render(row) : row[column.key]}
</td>
))}
</tr>
);
})}
</tbody>
</table>
</div>
)}
</div>
);
}
|