Agent skill
animated-focus
This document captures learnings from fixing keyboard navigation issues when floating components (Select, DropdownMenu, Popover) have CSS open/close animations.
Install this agent skill to your Project
npx add-skill https://github.com/aiskillstore/marketplace/tree/main/skills/andrehogberg/animated-focus
SKILL.md
Focus Management with CSS Animations
This document captures learnings from fixing keyboard navigation issues when floating components (Select, DropdownMenu, Popover) have CSS open/close animations.
The Problem
When floating content elements have CSS animations that start at opacity: 0 (like Tailwind's animate-in fade-in-0), the browser may reject element.focus() calls because the element is invisible.
Symptoms
- Component opens correctly with mouse clicks
- Keyboard navigation (arrow keys, Escape) doesn't work after opening with keyboard
- Works fine in demos without animation classes
- Broken in demos with animation classes
Root Cause
- CSS animations like
fade-in-0start the element atopacity: 0 - When
focus()is called immediately after render, the element is still invisible - Browser rejects focus on invisible elements
- Focus stays on the trigger button instead of moving to content
- Keyboard events go to trigger (which does nothing when open) instead of content
Evidence from Console Debugging
// After opening select, keyboard events go to trigger, not content:
Document keydown: ArrowDown Target: <button role="combobox" ...>
// Active element is trigger, not content:
Active: BUTTON summit-select-...-trigger
The Solution
Implement a retry mechanism for focus that allows the animation to progress past opacity: 0 before giving up.
JavaScript Implementation
// src/SummitUI/Scripts/floating.js
/**
* Focus an element with retry mechanism for animated elements.
* Elements with CSS animations starting at opacity:0 may reject focus initially.
* This retries focus up to 5 times with 20ms delays to allow the animation
* to progress past the invisible state.
* @param {HTMLElement} element - Element to focus
*/
export function focusElement(element) {
if (!element) return;
function tryFocus(attempts) {
element.focus();
// If focus didn't succeed and we have attempts left, retry
if (document.activeElement !== element && attempts > 0) {
setTimeout(() => tryFocus(attempts - 1), 20);
}
}
// First attempt after one frame to let CSS apply
requestAnimationFrame(() => tryFocus(5));
}
Key Points
- Use
requestAnimationFramefirst - Ensures CSS has been applied before attempting focus - Check
document.activeElement- Verify if focus actually succeeded - Retry with delays - 20ms intervals allow animation to progress
- Limited attempts - 5 retries = 100ms max wait, enough for typical animations
- Apply to all focus functions - Both
focusElement(element)andfocusElementById(id)need this pattern
Functions Updated
floating.js:focusElement(element)- Used by SelectContentfloating.js:focusElementById(elementId)- Used by DropdownMenuContent
Testing
Added "With Animations" sections to test demo pages and corresponding Playwright tests.
CSS Animation Classes for Testing
/* tests/SummitUI.Tests.Manual/SummitUI.Tests.Manual/wwwroot/app.css */
@keyframes fadeInZoomIn {
from {
opacity: 0;
transform: scale(0.95);
}
to {
opacity: 1;
transform: scale(1);
}
}
@keyframes fadeOutZoomOut {
from {
opacity: 1;
transform: scale(1);
}
to {
opacity: 0;
transform: scale(0.95);
}
}
.animated-content[data-state="open"] {
animation: fadeInZoomIn 150ms ease-out forwards;
}
.animated-content[data-state="closed"] {
animation: fadeOutZoomOut 150ms ease-in forwards;
}
Test Cases
For each component (Select, DropdownMenu, Popover):
| Test | What It Verifies |
|---|---|
Animated*_ShouldOpen_OnEnterKey |
Opens with keyboard when animations present |
Animated*_ShouldNavigate_WithArrowKeys |
Arrow keys work after animated open |
Animated*_ShouldSelect/Activate_OnEnterKey |
Can select/activate item after animated open |
Animated*_ShouldClose_OnEscape |
Escape triggers close animation |
Running Animation Tests
dotnet run --project tests/SummitUI.Tests.Playwright -- --treenode-filter '/*/*/*/Animated*'
Files Involved
| File | Purpose |
|---|---|
src/SummitUI/Scripts/floating.js |
Contains focusElement and focusElementById functions |
src/SummitUI/Components/Select/SelectContent.cs |
Calls FocusElementAsync on open |
src/SummitUI/Components/DropdownMenu/DropdownMenuContent.cs |
Calls FocusElementByIdAsync for menu items |
src/SummitUI/Components/Popover/PopoverContent.cs |
Manages focus for popover content |
Alternative Approaches Considered
- Longer initial delay - Could use 50-100ms delay before first focus attempt, but adds noticeable lag
- Focus before animation starts - Would require changes to render order, complex
- Disable animation during focus - Would cause visual glitch
- CSS
visibilityinstead ofopacity- Would require changes to how animations are authored
The retry mechanism was chosen because it:
- Works with any animation duration
- Doesn't require changes to CSS authoring
- Has minimal performance impact
- Fails gracefully if focus never succeeds
Related Patterns
This pattern is similar to how bits-ui handles animated presence in Svelte components. The key insight is that DOM operations (like focus) may need to wait for CSS animations to reach a focusable state.
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
perigon-backend
Perigon ASP.NET Core + EF Core + Aspire conventions
perigon-agent
Pointers for Copilot/agents to apply Perigon conventions
perigon-angular
Angular 21+ standalone/Material/signal conventions for Perigon WebApp
fastapi-mastery
Comprehensive FastAPI development skill covering REST API creation, routing, request/response handling, validation, authentication, database integration, middleware, and deployment. Use when working with FastAPI projects, building APIs, implementing CRUD operations, setting up authentication/authorization, integrating databases (SQL/NoSQL), adding middleware, handling WebSockets, or deploying FastAPI applications. Triggered by requests involving .py files with FastAPI code, API endpoint creation, Pydantic models, or FastAPI-specific features.
context7-efficient
Token-efficient library documentation fetcher using Context7 MCP with 86.8% token savings through intelligent shell pipeline filtering. Fetches code examples, API references, and best practices for JavaScript, Python, Go, Rust, and other libraries. Use when users ask about library documentation, need code examples, want API usage patterns, are learning a new framework, need syntax reference, or troubleshooting with library-specific information. Triggers include questions like "Show me React hooks", "How do I use Prisma", "What's the Next.js routing syntax", or any request for library/framework documentation.
browser-use
Browser automation using Playwright MCP. Navigate websites, fill forms, click elements, take screenshots, and extract data. Use when tasks require web browsing, form submission, web scraping, UI testing, or any browser interaction.
Didn't find tool you were looking for?