Skip to main content
Beta: Front-End Checklist is currently in beta. Some issues are still being fixed. Thanks for your patience.

Manage focus during dynamic interactions

Focus is programmatically managed during dynamic interactions like modals, page transitions, and content updates.

Utilities
Quick take
Typical fix time 25 min
  • Move focus to modal when opened, return to trigger when closed
  • Focus new content after SPA navigation or dynamic updates
  • Use tabindex='-1' to make non-interactive elements focusable
  • Never lose focus to an unexpected location
Why it matters: Poor focus management leaves keyboard users stranded—they can't find new content, get trapped in closed modals, or lose their place entirely after interactions.

Rule Details

Focus management ensures keyboard and screen reader users can follow dynamic content changes and interactions.

Code Example

import { useEffect, useRef } from 'react'
 
function Modal({ isOpen, onClose, children }) {
  const modalRef = useRef<HTMLDivElement>(null)
  const triggerRef = useRef<HTMLElement | null>(null)
 
  useEffect(() => {
    if (isOpen) {
      // Store the element that opened the modal
      triggerRef.current = document.activeElement as HTMLElement
      // Focus the modal
      modalRef.current?.focus()
    } else if (triggerRef.current) {
      // Return focus when closing
      triggerRef.current.focus()
    }
  }, [isOpen])
 
  if (!isOpen) return null
 
  return (
    <div
      ref={modalRef}
      role="dialog"
      aria-modal="true"
      tabIndex={-1}
    >
      {children}
      <button onClick={onClose}>Close</button>
    </div>
  )
}

Why It Matters

Poor focus management leaves keyboard users stranded—they can't find new content, get trapped in closed modals, or lose their place entirely after interactions.

Key Principles

ScenarioFocus Should Move To
Modal opensFirst focusable element in modal
Modal closesElement that triggered the modal
SPA navigationMain content heading or container
Content deletedPrevious/next item or parent container
Form submittedSuccess message or error summary

SPA Navigation Focus

import { useEffect, useRef } from 'react'
import { useLocation } from 'react-router-dom'
 
function MainContent({ children }) {
  const mainRef = useRef<HTMLElement>(null)
  const location = useLocation()
 
  useEffect(() => {
    // Focus main content on navigation
    mainRef.current?.focus()
  }, [location.pathname])
 
  return (
    <main ref={mainRef} tabIndex={-1}>
      {children}
    </main>
  )
}

Dynamic Content Updates

function TodoList({ todos, onDelete }) {
  const listRef = useRef<HTMLUListElement>(null)
  const [deletedIndex, setDeletedIndex] = useState<number | null>(null)
 
  const handleDelete = (index: number) => {
    setDeletedIndex(index)
    onDelete(index)
  }
 
  useEffect(() => {
    if (deletedIndex !== null) {
      // Focus next item, or previous, or the list itself
      const items = listRef.current?.querySelectorAll('button')
      const nextItem = items?.[deletedIndex] || items?.[deletedIndex - 1]
 
      if (nextItem) {
        nextItem.focus()
      } else {
        listRef.current?.focus()
      }
      setDeletedIndex(null)
    }
  }, [deletedIndex, todos])
 
  return (
    <ul ref={listRef} tabIndex={-1}>
      {todos.map((todo, index) => (
        <li key={todo.id}>
          {todo.text}
          <button onClick={() => handleDelete(index)}>Delete</button>
        </li>
      ))}
    </ul>
  )
}

Form Submission Focus

function ContactForm() {
  const [status, setStatus] = useState<'idle' | 'success' | 'error'>('idle')
  const statusRef = useRef<HTMLDivElement>(null)
 
  const handleSubmit = async (e: FormEvent) => {
    e.preventDefault()
    try {
      await submitForm()
      setStatus('success')
    } catch {
      setStatus('error')
    }
  }
 
  useEffect(() => {
    if (status !== 'idle') {
      statusRef.current?.focus()
    }
  }, [status])
 
  return (
    <form onSubmit={handleSubmit}>
      {/* Form fields */}
 
      {status !== 'idle' && (
        <div
          ref={statusRef}
          tabIndex={-1}
          role={status === 'error' ? 'alert' : 'status'}
        >
          {status === 'success' ? 'Message sent!' : 'Please fix errors above'}
        </div>
      )}
 
      <button type="submit">Send</button>
    </form>
  )
}

Exceptions

  • Temporary or intentionally inert UI can be removed from the focus order, but only when the same state is also communicated clearly to assistive technology users.
  • A focus-management issue should be evaluated in the rendered interaction, not only from static markup, because route changes, overlays, and JS timing can change the real behavior.
  • If a component is both unlabeled and focus-broken, fix the stronger user-facing orientation problem first rather than reporting multiple secondary symptoms.

Standards

  • Align the implementation with W3C WAI: WCAG Overview and verify the rendered experience, not only the source code.
  • Align the implementation with MDN: Accessibility and verify the rendered experience, not only the source code.

Verification

Automated Checks

  • Use browser accessibility tooling, axe, Lighthouse, or equivalent automated checks against a representative rendered state.

Manual Checks

  • Open modal with keyboard → focus should be inside modal
  • Close modal → focus should return to trigger element
  • Navigate SPA routes → focus should move to new content
  • Delete list items → focus should remain logical

Use with AI

Copy these prompts to use with your AI assistant, or install the MCP server to use directly from Claude, Cursor, or Windsurf.

Check

Verify implementation

Verify that focus is properly managed when opening/closing modals, navigating between views, or updating content dynamically.

Fix

Auto-fix issues

Implement proper focus management using tabindex, focus(), and focus trapping for modal dialogs.

Explain

Learn more

Explain how proper focus management ensures keyboard and screen reader users can navigate dynamic interfaces effectively.

Review

Code review

Review the rendered markup and interactive states that affect Manage focus during dynamic interactions. Flag exact elements, roles, labels, focus behavior, or keyboard interactions that violate the rule, and note how to verify the fix with browser accessibility tooling or assistive tech.

Sources

References used to support the guidance in this rule.

Further Reading

Tools and supplementary material for exploring the topic in more depth.

axe DevTools
deque.comTool

Rules that often go hand-in-hand with this one.

Enable keyboard navigation for all elements

All interactive elements are accessible via keyboard with logical focus order and hidden elements excluded from tab sequence.

Accessibility
Avoid autofocus on form fields

Form fields do not use the autofocus attribute which can disorient screen reader users and cause unexpected page behavior.

Accessibility
Include a skip navigation link

A skip navigation link is provided to allow keyboard users to bypass repetitive content and navigate directly to main content.

Accessibility
Ensure logical focus order

Tab focus order follows the visual layout and logical reading sequence of the page.

Accessibility

Was this rule helpful?

Your feedback helps improve rule quality. This stays internal for now.

Loading feedback...
0 / 385