2025-08-20 04:12:49 -06:00

153 lines
5.0 KiB
TypeScript

'use client';
import * as React from 'react';
import {
ChevronLeft,
ChevronRight,
Ban,
X,
Command,
IdCard,
} from 'lucide-react';
import { SlidingNumber } from '@/registry/text/sliding-number';
import { motion, type Variants, type Transition } from 'motion/react';
const TOTAL_PAGES = 10;
const BUTTON_MOTION_CONFIG = {
initial: 'rest',
whileHover: 'hover',
whileTap: 'tap',
variants: {
rest: { maxWidth: '40px' },
hover: {
maxWidth: '140px',
transition: { type: 'spring', stiffness: 200, damping: 35, delay: 0.15 },
},
tap: { scale: 0.95 },
},
transition: { type: 'spring', stiffness: 250, damping: 25 },
} as const;
const LABEL_VARIANTS: Variants = {
rest: { opacity: 0, x: 4 },
hover: { opacity: 1, x: 0, visibility: 'visible' },
tap: { opacity: 1, x: 0, visibility: 'visible' },
};
const LABEL_TRANSITION: Transition = {
type: 'spring',
stiffness: 200,
damping: 25,
};
function ManagementBar() {
const [currentPage, setCurrentPage] = React.useState(1);
const handlePrevPage = React.useCallback(() => {
if (currentPage > 1) setCurrentPage(currentPage - 1);
}, [currentPage]);
const handleNextPage = React.useCallback(() => {
if (currentPage < TOTAL_PAGES) setCurrentPage(currentPage + 1);
}, [currentPage]);
return (
<div className="flex w-fit flex-wrap items-center gap-y-2 rounded-2xl border border-border bg-background p-2 shadow-lg">
<div className="mx-auto flex shrink-0 items-center">
<button
disabled={currentPage === 1}
className="p-1 text-muted-foreground transition-colors hover:text-foreground disabled:text-muted-foreground/30 disabled:hover:text-muted-foreground/30"
onClick={handlePrevPage}
>
<ChevronLeft size={20} />
</button>
<div className="mx-2 flex items-center space-x-1 text-sm tabular-nums">
<SlidingNumber
className="text-foreground"
padStart
number={currentPage}
/>
<span className="text-muted-foreground">/ {TOTAL_PAGES}</span>
</div>
<button
disabled={currentPage === TOTAL_PAGES}
className="p-1 text-muted-foreground transition-colors hover:text-foreground disabled:text-muted-foreground/30 disabled:hover:text-muted-foreground/30"
onClick={handleNextPage}
>
<ChevronRight size={20} />
</button>
</div>
<div className="mx-3 h-6 w-px bg-border rounded-full" />
<motion.div
layout
layoutRoot
className="mx-auto flex flex-wrap space-x-2 sm:flex-nowrap"
>
<motion.button
{...BUTTON_MOTION_CONFIG}
className="flex h-10 items-center space-x-2 overflow-hidden whitespace-nowrap rounded-lg bg-neutral-200/60 dark:bg-neutral-600/80 px-2.5 py-2 text-neutral-600 dark:text-neutral-200"
aria-label="Blacklist"
>
<Ban size={20} className="shrink-0" />
<motion.span
variants={LABEL_VARIANTS}
transition={LABEL_TRANSITION}
className="invisible text-sm"
>
Blacklist
</motion.span>
</motion.button>
<motion.button
{...BUTTON_MOTION_CONFIG}
className="flex h-10 items-center space-x-2 overflow-hidden whitespace-nowrap rounded-lg bg-red-200/60 dark:bg-red-800/80 px-2.5 py-2 text-red-600 dark:text-red-300"
aria-label="Reject"
>
<X size={20} className="shrink-0" />
<motion.span
variants={LABEL_VARIANTS}
transition={LABEL_TRANSITION}
className="invisible text-sm"
>
Reject
</motion.span>
</motion.button>
<motion.button
{...BUTTON_MOTION_CONFIG}
className="flex h-10 items-center space-x-2 overflow-hidden whitespace-nowrap rounded-lg bg-green-200/60 dark:bg-green-800/80 px-2.5 py-2 text-green-600 dark:text-green-300"
aria-label="Hire"
>
<IdCard size={20} className="shrink-0" />
<motion.span
variants={LABEL_VARIANTS}
transition={LABEL_TRANSITION}
className="invisible text-sm"
>
Hire
</motion.span>
</motion.button>
</motion.div>
<div className="mx-3 hidden h-6 w-px bg-border sm:block rounded-full" />
<motion.button
whileTap={{ scale: 0.975 }}
className="flex w-full h-10 text-sm cursor-pointer items-center justify-center rounded-lg bg-teal-500 dark:bg-teal-600/80 px-3 py-2 text-white transition-colors duration-300 dark:hover:bg-teal-800 hover:bg-teal-600 sm:w-auto"
>
<span className="mr-1 text-neutral-200">Move to:</span>
<span>Interview I</span>
<div className="mx-3 h-5 w-px bg-white/40 rounded-full" />
<div className="flex items-center gap-1 rounded-md bg-white/20 px-1.5 py-0.5 -mr-1">
<Command size={14} />E
</div>
</motion.button>
</div>
);
}
export { ManagementBar };