Loading...
Loading...
Accessibility refactoring specialist. Automatically fixes accessibility issues across multiple files. Performs complex refactoring like extracting accessible components, restructuring markup, and implementing proper ARIA patterns.
npx skill4agent add accesslint/claude-marketplace refactoraccesslint:contrast-checkerpath/to/file.tsx<div onClick={handleClick}>Click me</div><button onClick={handleClick} aria-label="Submit form">
Click me
</button>src/components/Modal.tsx// Before
export function Modal({ isOpen, children, onClose }) {
if (!isOpen) return null;
return (
<div className="modal-overlay" onClick={onClose}>
<div className="modal-content">
{children}
</div>
</div>
);
}
// After
import { useEffect, useRef } from 'react';
import FocusTrap from 'focus-trap-react';
export function Modal({ isOpen, children, onClose, title, titleId = 'modal-title' }) {
const previousFocusRef = useRef<HTMLElement | null>(null);
useEffect(() => {
if (isOpen) {
// Store the currently focused element
previousFocusRef.current = document.activeElement as HTMLElement;
} else if (previousFocusRef.current) {
// Return focus when modal closes
previousFocusRef.current.focus();
}
}, [isOpen]);
useEffect(() => {
const handleEscape = (e: KeyboardEvent) => {
if (e.key === 'Escape') {
onClose();
}
};
if (isOpen) {
document.addEventListener('keydown', handleEscape);
return () => document.removeEventListener('keydown', handleEscape);
}
}, [isOpen, onClose]);
if (!isOpen) return null;
return (
<div className="modal-overlay" onClick={onClose}>
<FocusTrap>
<div
className="modal-content"
role="dialog"
aria-modal="true"
aria-labelledby={titleId}
onClick={(e) => e.stopPropagation()}
>
{children}
</div>
</FocusTrap>
</div>
);
}