3d Cursor Effect
An interactive React component that adds a dynamic bubble effect, visually tracking cursor movement in real time.
An interactive React component that adds a dynamic bubble effect, visually tracking cursor movement in real time.
1'use client';23import { useState, useEffect } from 'react';4import { useMouse } from '@/hooks/use-mouse';56const ThreeDCursor = () => {7const [mouseState, ref] = useMouse();8const [rotation, setRotation] = useState({ x: 0, y: 0 });9const [isHovering, setIsHovering] = useState(false);1011useEffect(() => {12if (mouseState.x && mouseState.y) {13const centerX = window.innerWidth / 2;14const centerY = window.innerHeight / 2;1516const rotateX = ((mouseState.y - centerY) / centerY) * 30;17const rotateY = ((mouseState.x - centerX) / centerX) * 30;1819setRotation({ x: rotateX, y: rotateY });20}21}, [mouseState.x, mouseState.y]);2223return (24<div className='relative w-full h-full overflow-hidden' ref={ref}>25{mouseState.x !== null && mouseState.y !== null && (26<div27className='fixed pointer-events-none z-50 transition-transform duration-100'28style={{29left: mouseState.x,30top: mouseState.y,31transform: `translate(-50%, -50%) perspective(1000px)32rotateX(${rotation.x}deg) rotateY(${rotation.y}deg)33scale(${isHovering ? 1.5 : 1})`,34}}35>36{/* 3D Cursor Structure */}37<div className='relative w-12 h-12'>38{/* Front face */}39<div40className='absolute inset-0 bg-indigo-500 rounded-lg mix-blend-screen'41style={{42transform: 'translateZ(6px)',43boxShadow: '0 0 20px rgba(99, 102, 241, 0.5)',44}}45/>4647{/* Back face */}48<div49className='absolute inset-0 bg-indigo-700 rounded-lg mix-blend-screen'50style={{51transform: 'translateZ(-6px)',52}}53/>5455{/* Side faces */}56{[...Array(4)].map((_, index) => (57<div58key={index}59className='absolute inset-0 bg-indigo-600 mix-blend-screen'60style={{61transform: `rotateY(${index * 90}deg) translateZ(6px)`,62width: '12px',63left: index % 2 === 0 ? 0 : 'auto',64right: index % 2 === 1 ? 0 : 'auto',65}}66/>67))}68</div>6970{/* Shadow */}71<div72className='absolute rounded-full bg-black/30 blur-md'73style={{74width: '48px',75height: '8px',76top: '100%',77left: '50%',78transform: 'translate(-50%, -4px) rotateX(90deg)',79}}80/>81</div>82)}8384<div className='flex items-center justify-center h-full gap-8'>85<button86className='px-8 py-4 bg-indigo-600 text-white rounded-lg transition-all duration-300'87style={{88transform: `perspective(1000px)89rotateX(${-rotation.x * 0.5}deg)90rotateY(${-rotation.y * 0.5}deg)`,91}}92onMouseEnter={() => setIsHovering(true)}93onMouseLeave={() => setIsHovering(false)}94>953D Hover Effect96</button>97</div>98</div>99);100};101102export default ThreeDCursor;103
1'use client';2import { type RefObject, useLayoutEffect, useRef, useState } from 'react';34interface MouseState {5x: number | null;6y: number | null;7elementX: number | null;8elementY: number | null;9elementPositionX: number | null;10elementPositionY: number | null;11}1213export function useMouse(): [MouseState, RefObject<HTMLDivElement>] {14const [state, setState] = useState<MouseState>({15x: null,16y: null,17elementX: null,18elementY: null,19elementPositionX: null,20elementPositionY: null,21});2223const ref = useRef<HTMLDivElement | null>(null);2425useLayoutEffect(() => {26const handleMouseMove = (event: MouseEvent) => {27const newState: Partial<MouseState> = {28x: event.pageX,29y: event.pageY,30};3132if (ref.current instanceof Element) {33const { left, top } = ref.current.getBoundingClientRect();34const elementPositionX = left + window.scrollX;35const elementPositionY = top + window.scrollY;36const elementX = event.pageX - elementPositionX;37const elementY = event.pageY - elementPositionY;3839newState.elementX = elementX;40newState.elementY = elementY;41newState.elementPositionX = elementPositionX;42newState.elementPositionY = elementPositionY;43}4445setState((s) => ({46...s,47...newState,48}));49};5051document.addEventListener('mousemove', handleMouseMove);5253return () => {54document.removeEventListener('mousemove', handleMouseMove);55};56}, []);5758return [state, ref];59}