An animated hamburger menu built with Framer Motion
It will create a new file Hamburger.tsx inside the components/menu/Hamburger.tsx directory.
Hamburger.tsx
components/menu/Hamburger.tsx
mkdir -p components/menu && touch components/menu/Hamburger.tsx
Open the newly created file and paste the following code:
import { motion } from "framer-motion"; import { Dispatch, SetStateAction, useState } from "react"; import { cn } from "@/lib/utils"; interface HamburgerProps { isActive: boolean; setIsActive: Dispatch<SetStateAction<boolean>>; direction?: "left" | "right"; } const App = () => { const [isActive, setIsActive] = useState(false); return ( <div className="size-full center"> <Hamburger isActive={isActive} setIsActive={setIsActive} /> </div> ); }; const Hamburger = ({ isActive, setIsActive, direction = "right", }: HamburgerProps) => { return ( <motion.div className="h-5 w-8 flex-col flex justify-between cursor-pointer hover:opacity-85 relative" onClick={() => setIsActive((prev) => !prev)} > {Array.from({ length: 3 }).map((_, index) => ( <motion.div key={index} className={cn("h-1 bg-primary")} initial={false} animate={ index === 1 ? { x: isActive ? (direction === "left" ? "-100%" : "100%") : 0, opacity: isActive ? 0 : 1, } : index === 0 ? { y: isActive ? "150%" : 0, rotate: isActive ? 45 : 0, } : { y: isActive ? "-250%" : "0%", rotate: isActive ? -45 : 0, } } transition={{ duration: 0.3 }} style={{ width: "100%" }} /> ))} </motion.div> ); }; export default App;
import { useState } from "react"; import { Hamburger } from "@/components/menu/Hamburger"; export default function Example() { const [isActive, setIsActive] = useState(false); return ( <Hamburger isActive={isActive} setIsActive={setIsActive} direction="right" /> ); }
Built by Bossadi Zenith