Better UIBetter UI

Perspective Text

A 3D rotating text component with interactive perspective effects, customizable rotation speed, and smooth hover transitions.

Overview

The PerspectiveText component creates an interactive 3D text effect using Framer Motion. It continuously rotates along the Y-axis and dynamically responds to mouse movements, providing a depth illusion.

The rotation speed and behavior can be customized for a more engaging user experience.


Preview

3D Perspective

Usage

'use client'
 
import { PerspectiveText } from '@/components/text/perspective-text'
 
<PerspectiveText text="3D Perspective" />

Code

'use client'
 
import { useCallback, useEffect } from 'react'
 
import { motion, useAnimation } from 'motion/react'
 
import { cn } from '@/lib/utils'
 
interface PerspectiveTextProps {
  text: string
  className?: string
  rotationSpeed?: number
  reverseRotation?: boolean
  as?: React.ElementType
}
 
export function PerspectiveText({
  text,
  className,
  rotationSpeed = 4,
  reverseRotation = true,
  as: Component = 'div',
}: PerspectiveTextProps) {
  const MotionComponent = motion.create(Component)
 
  const controls = useAnimation()
 
  useEffect(() => {
    controls.start({
      rotateY: reverseRotation ? -360 : 360,
      transition: {
        duration: rotationSpeed,
        repeat: Number.POSITIVE_INFINITY,
        ease: 'linear',
      },
    })
  }, [controls, rotationSpeed, reverseRotation])
 
  const handleMouseEnter = () => {
    controls.stop()
  }
 
  const handleMouseMove = useCallback(
    (event: React.MouseEvent<HTMLDivElement>) => {
      const { clientX, clientY, currentTarget } = event
      const { left, top, width, height } = currentTarget.getBoundingClientRect()
      const x = (clientX - left) / width
      const y = (clientY - top) / height
 
      controls.start({
        rotateX: 20 - y * 40,
        rotateY: -20 + x * 40,
        textShadow: `${(x - 0.5) * 20}px ${(y - 0.5) * 20}px 10px rgba(0,0,0,0.3)`,
        transition: { type: 'spring', stiffness: 300, damping: 30 },
      })
    },
    [controls]
  )
 
  const handleMouseLeave = () => {
    controls.start({
      rotateX: 5,
      rotateY: 5,
      textShadow: '0px 0px 5px rgba(0,0,0,0.2)',
      transition: { duration: 1, ease: 'easeOut' },
    })
 
    setTimeout(() => {
      controls.start({
        rotateY: reverseRotation ? -360 : 360,
        transition: {
          duration: rotationSpeed,
          repeat: Number.POSITIVE_INFINITY,
          ease: 'linear',
        },
      })
    }, 500)
  }
 
  return (
    <div
      className={cn('cursor-pointer text-4xl font-bold text-white', className)}
      style={{ perspective: '1000px' }}
      onMouseEnter={handleMouseEnter}
      onMouseMove={handleMouseMove}
      onMouseLeave={handleMouseLeave}
      aria-label={text}
    >
      <MotionComponent
        animate={controls}
        initial={{ rotateX: 0, rotateY: 0 }}
        whileHover={{ scale: 1.1 }}
      >
        {text}
      </MotionComponent>
    </div>
  )
}
 

Examples

rotationSpeed

The rotationSpeed is used to change the Y-axis rotation speed.

Fast Rotation

Fast Rotation

Slow Rotation

Slow Rotation

reverseRotation

The reverseRotation is used to text rotate counterclockwise instead of clockwise.

Reverse Rotation

className

The className is used to custom styles using classes.

Stylized Text

Props

PropTypeDefault
text
string
-
className
string
-
rotationSpeed
number
4
reverseRotation
boolean
false
as
ElementType<any, keyof IntrinsicElements>
"div"

On this page