91 lines
2.8 KiB
TypeScript
91 lines
2.8 KiB
TypeScript
import * as React from "react"
|
|
import { motion } from "framer-motion"
|
|
import { cn } from "@/lib/utils"
|
|
import { LucideIcon } from "lucide-react"
|
|
|
|
interface DockProps {
|
|
className?: string
|
|
items: {
|
|
icon: LucideIcon
|
|
label: string
|
|
onClick?: () => void
|
|
}[]
|
|
}
|
|
|
|
interface DockIconButtonProps {
|
|
icon: LucideIcon
|
|
label: string
|
|
onClick?: () => void
|
|
className?: string
|
|
}
|
|
|
|
const floatingAnimation = {
|
|
initial: { x: 0 },
|
|
animate: {
|
|
x: [-2, 2, -2],
|
|
transition: {
|
|
duration: 4,
|
|
repeat: Infinity,
|
|
ease: "easeInOut" as const
|
|
}
|
|
}
|
|
}
|
|
|
|
const DockIconButton = React.forwardRef<HTMLButtonElement, DockIconButtonProps>(
|
|
({ icon: Icon, label, onClick, className }, ref) => {
|
|
return (
|
|
<motion.button
|
|
ref={ref}
|
|
whileHover={{ scale: 1.1, x: -2 }}
|
|
whileTap={{ scale: 0.95 }}
|
|
onClick={onClick}
|
|
className={cn(
|
|
"relative group p-3 rounded-lg",
|
|
"hover:bg-secondary transition-colors",
|
|
className
|
|
)}
|
|
>
|
|
<Icon className="w-5 h-5 text-foreground" />
|
|
<span className={cn(
|
|
"absolute -left-16 top-1/2 -translate-y-1/2",
|
|
"px-2 py-1 rounded text-xs",
|
|
"bg-popover text-popover-foreground",
|
|
"opacity-0 group-hover:opacity-100",
|
|
"transition-opacity whitespace-nowrap pointer-events-none"
|
|
)}>
|
|
{label}
|
|
</span>
|
|
</motion.button>
|
|
)
|
|
}
|
|
)
|
|
DockIconButton.displayName = "DockIconButton"
|
|
|
|
const Dock = React.forwardRef<HTMLDivElement, DockProps>(
|
|
({ items, className }, ref) => {
|
|
return (
|
|
<div ref={ref} className={cn("w-16 h-full flex items-center justify-center p-2", className)}>
|
|
<div className="w-16 h-full max-h-4xl rounded-2xl flex items-center justify-center relative">
|
|
<motion.div
|
|
initial="initial"
|
|
animate="animate"
|
|
variants={floatingAnimation}
|
|
className={cn(
|
|
"flex flex-col items-center gap-1 p-2 rounded-2xl",
|
|
"backdrop-blur-lg border shadow-lg",
|
|
"bg-background/90 border-border",
|
|
"hover:shadow-xl transition-shadow duration-300"
|
|
)}
|
|
>
|
|
{items.map((item) => (
|
|
<DockIconButton key={item.label} {...item} />
|
|
))}
|
|
</motion.div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
)
|
|
Dock.displayName = "Dock"
|
|
|
|
export { Dock } |