mirror of
https://github.com/arc53/DocsGPT.git
synced 2025-11-29 16:43:16 +00:00
200 lines
7.3 KiB
TypeScript
200 lines
7.3 KiB
TypeScript
import { useState, useEffect } from 'react';
|
|
|
|
interface SkeletonLoaderProps {
|
|
count?: number;
|
|
component?:
|
|
| 'default'
|
|
| 'analysis'
|
|
| 'logs'
|
|
| 'table'
|
|
| 'chatbot'
|
|
| 'dropdown';
|
|
}
|
|
|
|
const SkeletonLoader: React.FC<SkeletonLoaderProps> = ({
|
|
count = 1,
|
|
component = 'default',
|
|
}) => {
|
|
const [skeletonCount, setSkeletonCount] = useState(count);
|
|
|
|
useEffect(() => {
|
|
const handleResize = () => {
|
|
const windowWidth = window.innerWidth;
|
|
|
|
if (windowWidth > 1024) {
|
|
setSkeletonCount(1);
|
|
} else if (windowWidth > 768) {
|
|
setSkeletonCount(count);
|
|
} else {
|
|
setSkeletonCount(Math.min(count, 2));
|
|
}
|
|
};
|
|
|
|
handleResize();
|
|
window.addEventListener('resize', handleResize);
|
|
|
|
return () => {
|
|
window.removeEventListener('resize', handleResize);
|
|
};
|
|
}, [count]);
|
|
|
|
const renderTable = () => (
|
|
<>
|
|
{[...Array(4)].map((_, idx) => (
|
|
<tr key={idx} className="animate-pulse">
|
|
<td className="w-[45%] px-4 py-4">
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="w-[20%] px-4 py-4">
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="w-[25%] px-4 py-4">
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="w-[10%] px-4 py-4">
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</>
|
|
);
|
|
|
|
const renderChatbot = () => (
|
|
<>
|
|
{[...Array(4)].map((_, idx) => (
|
|
<tr key={idx} className="animate-pulse">
|
|
<td className="p-2">
|
|
<div className="mx-auto h-4 w-3/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="p-2">
|
|
<div className="mx-auto h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="p-2">
|
|
<div className="mx-auto h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
<td className="p-2">
|
|
<div className="mx-auto h-4 w-8 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</>
|
|
);
|
|
|
|
const renderDropdown = () => (
|
|
<div className="animate-pulse">
|
|
<div className="mb-2 h-4 w-24 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="flex h-14 w-[360px] items-center justify-between rounded-3xl bg-gray-300 px-4 dark:bg-gray-600">
|
|
<div className="h-3 w-24 rounded bg-gray-400 dark:bg-gray-700"></div>
|
|
<div className="h-3 w-3 rounded bg-gray-400 dark:bg-gray-700"></div>
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
const renderLogs = () => (
|
|
<div className="w-full animate-pulse space-y-px">
|
|
{[...Array(8)].map((_, idx) => (
|
|
<div
|
|
key={idx}
|
|
className="flex w-full items-start p-2 hover:bg-[#F9F9F9] hover:dark:bg-dark-charcoal"
|
|
>
|
|
<div className="flex w-full items-center gap-2">
|
|
<div className="h-3 w-3 rounded-lg bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="flex w-full flex-row items-center gap-2">
|
|
<div className="h-3 w-[30%] rounded-lg bg-gray-300 dark:bg-gray-600 lg:w-52"></div>
|
|
<div className="h-3 w-[16%] rounded-lg bg-gray-300 dark:bg-gray-600 lg:w-28"></div>
|
|
<div className="h-3 w-[40%] rounded-lg bg-gray-300 dark:bg-gray-600 lg:w-64"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
);
|
|
|
|
const renderDefault = () => (
|
|
<>
|
|
{[...Array(skeletonCount)].map((_, idx) => (
|
|
<div
|
|
key={idx}
|
|
className={`p-6 ${
|
|
skeletonCount === 1 ? 'w-full' : 'w-60'
|
|
} animate-pulse rounded-3xl dark:bg-raisin-black`}
|
|
>
|
|
<div className="space-y-4">
|
|
<div>
|
|
<div className="mb-2 h-4 w-3/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-5/6 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-1/2 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-3/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
<div className="my-4 border-t border-gray-400 dark:border-gray-700"></div>
|
|
<div>
|
|
<div className="mb-2 h-4 w-2/3 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-1/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
<div className="my-4 border-t border-gray-400 dark:border-gray-700"></div>
|
|
<div>
|
|
<div className="mb-2 h-4 w-5/6 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-1/3 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-2/3 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
<div className="my-4 border-t border-gray-400 dark:border-gray-700"></div>
|
|
<div className="mb-2 h-4 w-3/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="mb-2 h-4 w-5/6 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</>
|
|
);
|
|
|
|
const renderAnalysis = () => (
|
|
<>
|
|
{[...Array(skeletonCount)].map((_, idx) => (
|
|
<div
|
|
key={idx}
|
|
className="w-full animate-pulse rounded-3xl p-6 dark:bg-raisin-black"
|
|
>
|
|
<div className="space-y-6">
|
|
<div className="space-y-2">
|
|
<div className="mb-4 h-4 w-1/3 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="grid grid-cols-6 items-end gap-2">
|
|
<div className="h-32 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-24 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-40 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-28 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-36 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-20 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
</div>
|
|
<div className="space-y-2">
|
|
<div className="mb-4 h-4 w-1/4 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-32 rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
<div className="grid grid-cols-2 gap-4">
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
<div className="h-4 w-full rounded bg-gray-300 dark:bg-gray-600"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</>
|
|
);
|
|
|
|
const componentMap = {
|
|
table: renderTable,
|
|
chatbot: renderChatbot,
|
|
dropdown: renderDropdown,
|
|
logs: renderLogs,
|
|
default: renderDefault,
|
|
analysis: renderAnalysis,
|
|
};
|
|
|
|
const render = componentMap[component] || componentMap.default;
|
|
|
|
return <>{render()}</>;
|
|
};
|
|
|
|
export default SkeletonLoader;
|