BadtzUI
    Beta
    Docs
    Shuffle Button

    Shuffle Button

    A dynamic button that shuffles the characters of its text when hovered over, creating an engaging animation.

    Loading...

    Installation

    Install dependencies

    npm install

    npm install clsx tailwind-merge

    Add utils file

    lib/utils.ts

    import { ClassValue, clsx } from "clsx";
    import { twMerge } from "tailwind-merge";
     
    export function cn(...inputs: ClassValue[]) {
      return twMerge(clsx(inputs));
    }

    Copy the source code

    shuffle-button.tsx

    "use client";
     
    import React, { useState, useEffect, useRef, ReactNode } from "react";
    import { cn } from "@/lib/utils";
     
    function shuffleChar(char: string): string {
      const characters = "abcdefghijklmnopqrstuvwxyz";
      return char === " "
        ? " "
        : characters[Math.floor(Math.random() * characters.length)];
    }
     
    interface ShuffleButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {
      children: string;
      className?: string;
      duration?: number;
    }
     
    export default function ShuffleButton({
      children,
      className,
      duration = 1,
      ...props
    }: ShuffleButtonProps) {
      const [shuffledText, setShuffledText] = useState(children);
      const [isHovering, setIsHovering] = useState(false);
      const intervals = useRef<NodeJS.Timeout[]>([]);
      const timeouts = useRef<NodeJS.Timeout[]>([]);
     
      useEffect(() => {
        const textArray = children.split("");
        const numberOfCharacters = textArray.filter((char) => char !== " ").length;
        const ABC = (duration * 500) / numberOfCharacters;
     
        if (isHovering) {
          textArray.forEach((char, index) => {
            if (char !== " ") {
              const intervalId = setInterval(() => {
                textArray[index] = shuffleChar(char);
                setShuffledText(textArray.join(""));
              }, 25);
              intervals.current.push(intervalId);
     
              const timeoutId = setTimeout(() => {
                clearInterval(intervalId);
                textArray[index] = children[index];
                setShuffledText(textArray.join(""));
              }, ABC * (index + 1));
              timeouts.current.push(timeoutId);
            }
          });
        } else {
          textArray.forEach((char, index) => {
            if (char !== " ") {
              const intervalId = setInterval(() => {
                textArray[numberOfCharacters - 1 - index] = shuffleChar(char);
                setShuffledText(textArray.join(""));
              }, 25);
              intervals.current.push(intervalId);
     
              const timeoutId = setTimeout(() => {
                clearInterval(intervalId);
                textArray[numberOfCharacters - 1 - index] =
                  children[numberOfCharacters - 1 - index];
                setShuffledText(textArray.join(""));
              }, ABC * (index + 1));
              timeouts.current.push(timeoutId);
            }
          });
        }
     
        return () => {
          intervals.current.forEach(clearInterval);
          timeouts.current.forEach(clearTimeout);
          intervals.current = [];
          timeouts.current = [];
        };
      }, [isHovering, children, duration]);
     
      return (
        <button
          className={cn(
            "font-mono bg-neutral-100 text-black dark:bg-neutral-900 dark:text-white h-10 px-4 py-2 inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
            className
          )}
          onMouseEnter={() => setIsHovering(true)}
          onMouseLeave={() => setIsHovering(false)}
          {...props}
        >
          {shuffledText}
        </button>
      );
    }

    Props

    PropTypeDescriptionDefault
    classNameStringThe class name for the component, allowing for custom styling.-
    durationNumberThe duration of the shuffle animation in seconds.1
    childrenReact.ReactNodeThe content inside the button, typically text, which will be shuffled on hover.undefined

    Credits

  1. This component is inspired by ChainGPT