Better UIBetter UI

Typing Simulator Text

A dynamic component that mimics real-time text typing and deletion effects.

Overview

The TypingSimulatorText component is designed to simulate typing effects for a text or a sequence of texts. It provides an animated text display where the characters are typed and then deleted in a looping or one-time animation cycle.

The component also offers flexibility in customizing the typing and deleting speeds, as well as the appearance of the cursor. You can use it for creating dynamic and engaging user interfaces where text is dynamically typed out.


Preview

Usage

'use client'
 
import { TypingSimulatorText } from '@/components/text/typing-simulator-text'
 
<TypingSimulatorText text="Typing Simulator" />

Code

'use client'
 
import { useState, useEffect, useRef, useCallback, useMemo } from 'react'
 
import { motion } from 'motion/react'
 
import { cn } from '@/lib/utils'
 
interface TypingSimulatorTextProps {
  text: string | string[]
  typingSpeed?: number
  deletingSpeed?: number
  pauseTime?: number
  cursorStyle?: string
  loop?: boolean
  className?: string
  as?: React.ElementType
}
 
export function TypingSimulatorText({
  text,
  typingSpeed = 100,
  deletingSpeed = 50,
  pauseTime = 1500,
  cursorStyle = '|',
  loop = false,
  className,
  as: Component = 'span',
}: TypingSimulatorTextProps) {
  const MotionComponent = motion.create(Component)
 
  const [displayText, setDisplayText] = useState('')
  const [textIndex, setTextIndex] = useState(0)
  const [isDeleting, setIsDeleting] = useState(false)
  const [isBlinking, setIsBlinking] = useState(true) // Track cursor state
 
  const timeoutRef = useRef<NodeJS.Timeout | null>(null)
 
  const texts = useMemo(
    () => (Array.isArray(text) ? text : [text]), // Normalize input
    [text]
  )
 
  const updateText = useCallback(() => {
    if (!loop && textIndex >= texts.length) return // Stop when done (if loop is false)
 
    const currentText = texts[textIndex]
 
    // Typing completed -> Pause before deleting
    if (!isDeleting && displayText === currentText) {
      // Pause before deleting
      if (loop || textIndex < texts.length - 1) {
        timeoutRef.current = setTimeout(() => {
          setIsDeleting(true)
          setIsBlinking(false) // Keep cursor steady while deleting
        }, pauseTime)
      }
      setIsBlinking(true) // Start blinking when idle
      return
    }
 
    // Deletion completed -> Move to next text
    if (isDeleting && displayText === '') {
      // Move to next word or stop if loop is false
      if (!loop && textIndex === texts.length - 1) return
      setIsDeleting(false)
      setTextIndex((prev) => (prev + 1) % texts.length)
      setIsBlinking(false) // Keep cursor steady while typing
      return
    }
 
    // Typing or deleting character-by-character
    const delta = isDeleting ? -1 : 1
    timeoutRef.current = setTimeout(
      () => {
        setDisplayText(currentText.substring(0, displayText.length + delta))
        setIsBlinking(false) // Cursor steady while typing/deleting
      },
      isDeleting ? deletingSpeed : typingSpeed
    )
  }, [
    deletingSpeed,
    displayText,
    isDeleting,
    loop,
    pauseTime,
    textIndex,
    texts,
    typingSpeed,
  ])
 
  useEffect(() => {
    // Clear previous timeout
    if (timeoutRef.current) clearTimeout(timeoutRef.current)
 
    updateText()
 
    return () => {
      if (timeoutRef.current) clearTimeout(timeoutRef.current)
    }
  }, [updateText, isDeleting, deletingSpeed, typingSpeed])
 
  return (
    <div
      className={cn('text-4xl font-bold', className)}
      role="region"
      aria-label="Typing simulator"
    >
      <span aria-live="polite">{displayText}</span>
      <MotionComponent
        animate={{ opacity: isBlinking ? [1, 0, 1] : 1 }} // Blinks only if when idle
        transition={{ duration: 0.8, repeat: isBlinking ? Infinity : 0 }} // No blinking when typing
        className="ml-0.5 text-white"
        aria-hidden="true"
      >
        {cursorStyle}
      </MotionComponent>
    </div>
  )
}
 

Examples

text

This text props defines the text that will be typed. It can either be a string or an array of strings.

typingSpeed

The typingSpeed prop controls how fast the text is typed out.

deletingSpeed

The deletingSpeed prop controls how fast the text is deleted.

pauseTime

The pauseTime prop controls how long the text will pause before deleting.

cursorStyle

The cursorStyle prop defines the character or style of the cursor.

loop

The loop prop, when set to true, will restart the typing animation after it completes.

className

The className prop allows you to add custom styling to the component.

Props

PropTypeDefault
text
string | string[]
-
typingSpeed
number
100
deletingSpeed
number
50
pauseTime
number
1500
cursorStyle
string
"|"
loop
boolean
false
className
string
-
as
ElementType<any, keyof IntrinsicElements>
"span"

On this page