Circular Slider

0

Installation

1

Install the following packages if you do not have it.

npm i framer-motion lucide-react
2

Copy and paste the following code into your project.

"use client";
import React, { useState, useEffect, useRef } from "react";
import { motion } from "framer-motion";

const CircularSlider = () => {
  const [isRotating, setIsRotating] = useState(false);
  const [value, setValue] = useState(0);
  const [isHovered, setIsHovered] = useState(false);
  const knobRef = useRef<HTMLDivElement>(null);
  const pointerRef = useRef<HTMLDivElement>(null);
  const circleRef = useRef<SVGCircleElement>(null);

  useEffect(() => {
    const handleMouseMove = (e: MouseEvent) => {
      if (
        isRotating &&
        knobRef.current &&
        pointerRef.current &&
        circleRef.current
      ) {
        const knobRect = knobRef.current.getBoundingClientRect();
        const knobX = knobRect.left + knobRect.width / 2;
        const knobY = knobRect.top + knobRect.height / 2;

        const deltaX = e.clientX - knobX;
        const deltaY = e.clientY - knobY;

        const angleRad = Math.atan2(deltaY, deltaX);
        const angleDeg = (angleRad * 180) / Math.PI;

        let rotationAngle = (angleDeg - 135 + 360) % 360;

        if (rotationAngle <= 270) {
          pointerRef.current.style.transform = `rotate(${
            rotationAngle - 45
          }deg)`;
          const progressPercent = rotationAngle / 270;
          circleRef.current.style.strokeDashoffset = `${
            880 - 660 * progressPercent
          }`;
          setValue(Math.round(progressPercent * 100));
        }
      }
    };

    const handleMouseUp = () => {
      setIsRotating(false);
    };

    document.addEventListener("mousemove", handleMouseMove);
    document.addEventListener("mouseup", handleMouseUp);

    return () => {
      document.removeEventListener("mousemove", handleMouseMove);
      document.removeEventListener("mouseup", handleMouseUp);
    };
  }, [isRotating]);

  const handleMouseDown = (e: React.MouseEvent) => {
    if (e.target instanceof Element && e.target.closest(".knob")) {
      setIsRotating(true);
    }
  };

  // Get color based on value
  const getColor = () => {
    if (value < 33) return "from-red-500 to-orange-500";
    if (value < 66) return "from-orange-500 to-yellow-500";
    return "from-green-400 to-emerald-500";
  };

  return (
    <div className="w-full h-96 relative overflow-hidden">
      <div className="w-full h-full flex items-center justify-center">
        <motion.div
          className="w-[300px] h-[300px] flex items-center justify-center relative select-none"
          whileHover={{ scale: 1.02 }}
          transition={{ type: "spring", stiffness: 300 }}
        >
          <div
            ref={knobRef}
            className="knob w-[220px] h-[220px] bg-gradient-to-br from-slate-300 to-slate-100 dark:from-slate-700 dark:to-slate-900 rounded-full absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 flex items-center justify-center cursor-pointer"
            onMouseDown={handleMouseDown}
            onMouseEnter={() => setIsHovered(true)}
            onMouseLeave={() => setIsHovered(false)}
          >
            <motion.div
              className="w-[180px] h-[180px] bg-gradient-to-br from-white to-slate-200 dark:from-slate-600 dark:to-slate-800 rounded-full shadow-lg flex items-center justify-center relative z-[1]"
              animate={{
                boxShadow: isHovered
                  ? "0 10px 30px -10px rgba(0,0,0,0.3), inset 0 1px 2px rgba(255,255,255,0.9)"
                  : "0 5px 15px -5px rgba(0,0,0,0.2), inset 0 1px 2px rgba(255,255,255,0.9)",
              }}
            >
              <motion.div
                className={`w-16 h-16 bg-gradient-to-br ${getColor()} rounded-full flex items-center justify-center shadow-lg`}
                animate={{ scale: isRotating ? 1.1 : 1 }}
              >
                <span className="text-xl font-bold text-white">{value}</span>
              </motion.div>
            </motion.div>

            {/* Modern pointer indicator */}
            <div
              ref={pointerRef}
              className="pointer absolute w-6 h-6 flex items-center justify-center"
              style={{
                top: "calc(50% - 12px)",
                left: 0,
                transformOrigin: "110px 12px",
                transform: "rotate(-45deg)",
              }}
            >
              <motion.div
                className="w-3 h-3 bg-gradient-to-br from-blue-500 to-blue-600 rounded-full"
                animate={{
                  scale: isRotating ? 1.2 : 1,
                  boxShadow: isRotating
                    ? "0 0 20px 2px rgba(59, 130, 246, 0.5)"
                    : "0 0 10px 1px rgba(59, 130, 246, 0.3)",
                }}
              />
            </div>
          </div>

          <svg className="w-[300px] h-[300px]">
            <circle
              cx="150"
              cy="150"
              r="140"
              className="fill-none stroke-slate-300/20 dark:stroke-slate-700/20 stroke-[15px]"
              style={{
                strokeDasharray: 660,
                transformOrigin: "center",
                transform: "rotate(135deg)",
                strokeLinecap: "round",
              }}
            />
            <circle
              ref={circleRef}
              cx="150"
              cy="150"
              r="140"
              className="fill-none stroke-[url(#gradient)] stroke-[15px]"
              style={{
                strokeDasharray: 880,
                strokeDashoffset: 880,
                transformOrigin: "center",
                transform: "rotate(135deg)",
                strokeLinecap: "round",
              }}
            />
            <defs>
              <linearGradient id="gradient" x1="0%" y1="0%" x2="100%" y2="100%">
                <stop offset="0%" style={{ stopColor: "rgb(239, 68, 68)" }} />
                <stop offset="50%" style={{ stopColor: "rgb(249, 115, 22)" }} />
                <stop
                  offset="100%"
                  style={{ stopColor: "rgb(16, 185, 129)" }}
                />
              </linearGradient>
            </defs>
          </svg>
        </motion.div>
      </div>
    </div>
  );
};

export default CircularSlider;
3

Implement the code as demonstrated in the preview