Vercel/src/admin/AdminNavClient.tsx
2025-02-25 23:31:19 -06:00

117 lines
3.6 KiB
TypeScript

'use client';
import LinkWithLoader from '@/components/LinkWithLoader';
import LinkWithStatus from '@/components/LinkWithStatus';
import Note from '@/components/Note';
import SiteGrid from '@/components/SiteGrid';
import Spinner from '@/components/Spinner';
import {
PATH_ADMIN_CONFIGURATION,
PATH_ADMIN_INSIGHTS,
checkPathPrefix,
isPathAdminInfo,
isPathTopLevelAdmin,
} from '@/app/paths';
import { useAppState } from '@/state/AppState';
import { clsx } from 'clsx/lite';
import { differenceInMinutes } from 'date-fns';
import { usePathname } from 'next/navigation';
import { useEffect, useMemo, useState } from 'react';
import { FaRegClock } from 'react-icons/fa';
import AdminAppInfoIcon from './AdminAppInfoIcon';
import AdminInfoNav from './AdminInfoNav';
// Updates considered recent if they occurred in past 5 minutes
const areTimesRecent = (dates: Date[]) => dates
.some(date => differenceInMinutes(new Date(), date) < 5);
export default function AdminNavClient({
items,
mostRecentPhotoUpdateTime,
includeInsights = true,
}: {
items: {
label: string,
href: string,
count: number,
}[]
mostRecentPhotoUpdateTime?: Date
includeInsights?: boolean
}) {
const pathname = usePathname();
const { adminUpdateTimes = [] } = useAppState();
const updateTimes = useMemo(() =>
(mostRecentPhotoUpdateTime ? [mostRecentPhotoUpdateTime] : [])
.concat(adminUpdateTimes)
, [mostRecentPhotoUpdateTime, adminUpdateTimes]);
const [hasRecentUpdates, setHasRecentUpdates] =
useState(areTimesRecent(updateTimes));
useEffect(() => {
// Check every 5 seconds if update times are recent
setHasRecentUpdates(areTimesRecent(updateTimes));
const interval = setInterval(() =>
setHasRecentUpdates(areTimesRecent(updateTimes))
, 5_000);
return () => clearInterval(interval);
}, [updateTimes]);
const shouldShowBanner = hasRecentUpdates && isPathTopLevelAdmin(pathname);
return (
<SiteGrid
contentMain={
<div className="space-y-5">
<div className={clsx(
'flex gap-2 pb-3',
'border-b border-gray-200 dark:border-gray-800',
)}>
<div className={clsx(
'flex gap-0.5 md:gap-1.5 -mx-1',
'grow overflow-x-auto',
)}>
{items.map(({ label, href, count }) =>
<LinkWithStatus
key={label}
href={href}
className={clsx(
'flex gap-0.5',
checkPathPrefix(pathname, href) ? 'font-bold' : 'text-dim',
'px-1 py-0.5 rounded-md',
)}
loadingClassName="bg-dim"
prefetch={false}
>
<span>{label}</span>
{count > 0 &&
<span>({count})</span>}
</LinkWithStatus>)}
</div>
<LinkWithLoader
href={includeInsights
? PATH_ADMIN_INSIGHTS
: PATH_ADMIN_CONFIGURATION}
className={isPathAdminInfo(pathname)
? 'font-bold'
: 'text-dim'}
loader={<Spinner className="translate-y-[-0.75px]" />}
>
<AdminAppInfoIcon />
</LinkWithLoader>
</div>
{shouldShowBanner &&
<Note icon={<FaRegClock className="shrink-0" />}>
Photo updates detectedthey may take several minutes to show up
for visitors
</Note>}
{isPathAdminInfo(pathname) &&
<AdminInfoNav {...{ includeInsights }} />}
</div>
}
/>
);
}