* Extract out ShareHover components * Refactor hover/category state * Rename photo query options types * Restore category count slice of app state * Streamline entity hover headers * Standardize swr keys * Suppress hover counts to years * Refine entity hover design * Make image hovers opt out
196 lines
5.9 KiB
TypeScript
196 lines
5.9 KiB
TypeScript
import Switcher from '@/components/Switcher';
|
|
import SwitcherItem from '@/components/SwitcherItem';
|
|
import IconFeed from '@/components/icons/IconFeed';
|
|
import IconGrid from '@/components/icons/IconGrid';
|
|
import {
|
|
doesPathOfferSort,
|
|
PATH_FEED_INFERRED,
|
|
PATH_GRID_INFERRED,
|
|
} from '@/app/paths';
|
|
import IconSearch from '../components/icons/IconSearch';
|
|
import { useAppState } from '@/app/AppState';
|
|
import {
|
|
GRID_HOMEPAGE_ENABLED,
|
|
SHOW_KEYBOARD_SHORTCUT_TOOLTIPS,
|
|
SHOW_SORT_CONTROL,
|
|
} from './config';
|
|
import AdminAppMenu from '@/admin/AdminAppMenu';
|
|
import Spinner from '@/components/Spinner';
|
|
import clsx from 'clsx/lite';
|
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
|
import useKeydownHandler from '@/utility/useKeydownHandler';
|
|
import { usePathname } from 'next/navigation';
|
|
import { KEY_COMMANDS } from '@/photo/key-commands';
|
|
import { useAppText } from '@/i18n/state/client';
|
|
import IconSort from '@/components/icons/IconSort';
|
|
import { getSortConfigFromPath } from '@/photo/db/sort-path';
|
|
import { motion } from 'framer-motion';
|
|
|
|
export type SwitcherSelection = 'feed' | 'grid' | 'admin';
|
|
|
|
const GAP_CLASS = 'mr-1.5 sm:mr-2';
|
|
|
|
export default function AppViewSwitcher({
|
|
currentSelection,
|
|
className,
|
|
animate = true,
|
|
}: {
|
|
currentSelection?: SwitcherSelection
|
|
className?: string
|
|
animate?: boolean
|
|
}) {
|
|
const pathname = usePathname();
|
|
|
|
const appText = useAppText();
|
|
|
|
const {
|
|
isUserSignedIn,
|
|
isUserSignedInEager,
|
|
setIsCommandKOpen,
|
|
invalidateSwr,
|
|
} = useAppState();
|
|
|
|
const showSortControl = SHOW_SORT_CONTROL && doesPathOfferSort(pathname);
|
|
const {
|
|
sortBy,
|
|
isAscending,
|
|
pathGrid,
|
|
pathFeed,
|
|
pathSort,
|
|
} = getSortConfigFromPath(pathname);
|
|
|
|
const hasLoadedRef = useRef(false);
|
|
useEffect(() => {
|
|
if (hasLoadedRef.current) {
|
|
// After initial load, invalidate cache every time sort changes
|
|
invalidateSwr?.();
|
|
}
|
|
hasLoadedRef.current = true;
|
|
}, [invalidateSwr, sortBy]);
|
|
|
|
const refHrefFeed = useRef<HTMLAnchorElement>(null);
|
|
const refHrefGrid = useRef<HTMLAnchorElement>(null);
|
|
|
|
const onKeyDown = useCallback((e: KeyboardEvent) => {
|
|
if (!e.metaKey) {
|
|
switch (e.key.toLocaleUpperCase()) {
|
|
case KEY_COMMANDS.feed:
|
|
if (pathname !== PATH_FEED_INFERRED) { refHrefFeed.current?.click(); }
|
|
break;
|
|
case KEY_COMMANDS.grid:
|
|
if (pathname !== PATH_GRID_INFERRED) { refHrefGrid.current?.click(); }
|
|
break;
|
|
case KEY_COMMANDS.admin:
|
|
if (isUserSignedIn) { setIsAdminMenuOpen(true); }
|
|
break;
|
|
}
|
|
}
|
|
}, [pathname, isUserSignedIn]);
|
|
useKeydownHandler({ onKeyDown });
|
|
|
|
const [isAdminMenuOpen, setIsAdminMenuOpen] = useState(false);
|
|
|
|
const renderItemFeed =
|
|
<SwitcherItem
|
|
icon={<IconFeed includeTitle={false} />}
|
|
href={pathFeed}
|
|
hrefRef={refHrefFeed}
|
|
active={currentSelection === 'feed'}
|
|
tooltip={{...SHOW_KEYBOARD_SHORTCUT_TOOLTIPS && {
|
|
content: appText.nav.feed,
|
|
keyCommand: KEY_COMMANDS.feed,
|
|
}}}
|
|
noPadding
|
|
/>;
|
|
|
|
const renderItemGrid =
|
|
<SwitcherItem
|
|
icon={<IconGrid includeTitle={false} />}
|
|
href={pathGrid}
|
|
hrefRef={refHrefGrid}
|
|
active={currentSelection === 'grid'}
|
|
tooltip={{...SHOW_KEYBOARD_SHORTCUT_TOOLTIPS && {
|
|
content: appText.nav.grid,
|
|
keyCommand: KEY_COMMANDS.grid,
|
|
}}}
|
|
noPadding
|
|
/>;
|
|
|
|
return (
|
|
<div className={clsx('flex', className)}>
|
|
<Switcher className={GAP_CLASS}>
|
|
{GRID_HOMEPAGE_ENABLED ? renderItemGrid : renderItemFeed}
|
|
{GRID_HOMEPAGE_ENABLED ? renderItemFeed : renderItemGrid}
|
|
{/* Show spinner if admin is suspected to be logged in */}
|
|
{(isUserSignedInEager && !isUserSignedIn) &&
|
|
<SwitcherItem
|
|
icon={<Spinner />}
|
|
isInteractive={false}
|
|
noPadding
|
|
tooltip={{
|
|
...!isAdminMenuOpen && SHOW_KEYBOARD_SHORTCUT_TOOLTIPS && {
|
|
content: appText.nav.admin,
|
|
keyCommand: KEY_COMMANDS.admin,
|
|
},
|
|
}}
|
|
/>}
|
|
{isUserSignedIn &&
|
|
<SwitcherItem
|
|
icon={<AdminAppMenu
|
|
isOpen={isAdminMenuOpen}
|
|
setIsOpen={setIsAdminMenuOpen}
|
|
/>}
|
|
tooltip={{
|
|
...!isAdminMenuOpen && SHOW_KEYBOARD_SHORTCUT_TOOLTIPS && {
|
|
content: appText.nav.admin,
|
|
keyCommand: KEY_COMMANDS.admin,
|
|
},
|
|
}}
|
|
noPadding
|
|
/>}
|
|
</Switcher>
|
|
{showSortControl &&
|
|
<motion.div
|
|
initial={animate ? { opacity: 0, scale: 0.5 } : false}
|
|
animate={{ opacity: 1, scale: 1 }}
|
|
exit={{ opacity: 0, scale: 0.5 }}
|
|
transition={{ duration: 0.2, ease: 'easeInOut' }}
|
|
className={GAP_CLASS}
|
|
>
|
|
<Switcher className="max-sm:hidden">
|
|
<SwitcherItem
|
|
href={pathSort}
|
|
icon={<IconSort
|
|
sort={isAscending ? 'asc' : 'desc'}
|
|
className="translate-x-[0.5px] translate-y-[1px]"
|
|
/>}
|
|
tooltip={{
|
|
content: isAscending
|
|
? appText.sort.newest
|
|
: appText.sort.oldest,
|
|
}}
|
|
/>
|
|
</Switcher>
|
|
</motion.div>}
|
|
<motion.div
|
|
// Conditional key necessary to halt/resume layout animations
|
|
key={animate ? 'search' : 'search-no-animate'}
|
|
layout={animate}
|
|
transition={{ duration: 0.2, ease: 'easeInOut' }}
|
|
>
|
|
<Switcher type="borderless">
|
|
<SwitcherItem
|
|
icon={<IconSearch includeTitle={false} />}
|
|
onClick={() => setIsCommandKOpen?.(true)}
|
|
tooltip={{...SHOW_KEYBOARD_SHORTCUT_TOOLTIPS && {
|
|
content: appText.nav.search,
|
|
keyCommandModifier: KEY_COMMANDS.search[0],
|
|
keyCommand: KEY_COMMANDS.search[1],
|
|
}}}
|
|
/>
|
|
</Switcher>
|
|
</motion.div>
|
|
</div>
|
|
);
|
|
}
|