Animated Theme Toggle

Installation

1

Install the following packages if you do not have it.

npm i framer-motion lucide-react next-themes
2

Copy and paste the following code into your project.

"use client";

import * as React from "react";
import { motion, AnimatePresence } from "framer-motion";
import { Moon, Sun } from "lucide-react";
import { useTheme, ThemeProvider } from "next-themes";

export function AnimatedThemeToggle() {
  const { theme, setTheme } = useTheme();
  const [mounted, setMounted] = React.useState(false);

  React.useEffect(() => {
    setMounted(true);
  }, []);

  const toggleTheme = () => {
    setTheme(theme === "light" ? "dark" : "light");
  };

  if (!mounted) {
    return null;
  }

  return (
    <ThemeProvider attribute="class" defaultTheme="system" enableSystem>
      <motion.button
        className="relative w-12 h-12 rounded-full bg-gradient-to-br from-yellow-400 to-orange-500 dark:from-indigo-600 dark:to-purple-800 p-1 overflow-hidden focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-white focus:ring-yellow-500 dark:focus:ring-offset-gray-900 dark:focus:ring-indigo-400"
        onClick={toggleTheme}
        aria-label={`Switch to ${theme === "light" ? "dark" : "light"} mode`}
        whileTap={{ scale: 0.95 }}
      >
        <AnimatePresence mode="wait" initial={false}>
          <motion.div
            key={theme}
            initial={{ y: -30, opacity: 0 }}
            animate={{ y: 0, opacity: 1 }}
            exit={{ y: 30, opacity: 0 }}
            transition={{ duration: 0.2 }}
            className="w-full h-full rounded-full bg-white dark:bg-gray-950 flex items-center justify-center"
          >
            {theme === "light" ? (
              <Sun className="w-6 h-6 text-yellow-500" />
            ) : (
              <Moon className="w-6 h-6 text-indigo-300" />
            )}
          </motion.div>
        </AnimatePresence>
      </motion.button>
    </ThemeProvider>
  );
}

export function TransitionEffect() {
  const { theme } = useTheme();
  const [prevTheme, setPrevTheme] = React.useState<string | undefined>(
    undefined
  );

  const [mounted, setMounted] = React.useState(false);

  React.useEffect(() => {
    setMounted(true);
  }, []);

  React.useEffect(() => {
    if (theme !== prevTheme) {
      setPrevTheme(theme);
    }
  }, [theme, prevTheme]);

  if (!mounted) {
    return null;
  }

  return (
    <AnimatePresence mode="wait">
      {theme !== prevTheme && (
        <motion.div
          key={theme}
          initial={{ opacity: 1 }}
          animate={{ opacity: 0 }}
          exit={{ opacity: 0 }}
          transition={{ duration: 0.5 }}
          className="fixed inset-0 z-50 flex items-center justify-center pointer-events-none"
        >
          <div className="w-full h-full bg-white dark:bg-gray-950 transition-colors duration-300" />
          <motion.div
            className="absolute"
            initial={{ scale: 0 }}
            animate={{ scale: 4 }}
            transition={{ duration: 0.5, ease: "easeInOut" }}
          >
            {theme === "dark" ? (
              <Moon className="w-16 h-16 text-indigo-300" />
            ) : (
              <Sun className="w-16 h-16 text-yellow-500" />
            )}
          </motion.div>
        </motion.div>
      )}
    </AnimatePresence>
  );
}
3

Implement the code as demonstrated in the preview