| 38 | * not with a mouse, touch, or other input methods. |
| 39 | */ |
| 40 | export function useFocusRing(props: AriaFocusRingProps = {}): FocusRingAria { |
| 41 | let {autoFocus = false, isTextInput, within} = props; |
| 42 | let state = useRef({ |
| 43 | isFocused: false, |
| 44 | isFocusVisible: autoFocus || isFocusVisible() |
| 45 | }); |
| 46 | let [isFocused, setFocused] = useState(false); |
| 47 | let [isFocusVisibleState, setFocusVisible] = useState( |
| 48 | () => state.current.isFocused && state.current.isFocusVisible |
| 49 | ); |
| 50 | |
| 51 | let updateState = useCallback( |
| 52 | () => setFocusVisible(state.current.isFocused && state.current.isFocusVisible), |
| 53 | [] |
| 54 | ); |
| 55 | |
| 56 | let onFocusChange = useCallback( |
| 57 | isFocused => { |
| 58 | state.current.isFocused = isFocused; |
| 59 | state.current.isFocusVisible = isFocusVisible(); |
| 60 | setFocused(isFocused); |
| 61 | updateState(); |
| 62 | }, |
| 63 | [updateState] |
| 64 | ); |
| 65 | |
| 66 | useFocusVisibleListener( |
| 67 | isFocusVisible => { |
| 68 | state.current.isFocusVisible = isFocusVisible; |
| 69 | updateState(); |
| 70 | }, |
| 71 | [isTextInput, isFocused], |
| 72 | {enabled: isFocused, isTextInput} |
| 73 | ); |
| 74 | |
| 75 | let {focusProps} = useFocus({ |
| 76 | isDisabled: within, |
| 77 | onFocusChange |
| 78 | }); |
| 79 | |
| 80 | let {focusWithinProps} = useFocusWithin({ |
| 81 | isDisabled: !within, |
| 82 | onFocusWithinChange: onFocusChange |
| 83 | }); |
| 84 | |
| 85 | return { |
| 86 | isFocused, |
| 87 | isFocusVisible: isFocusVisibleState, |
| 88 | focusProps: within ? focusWithinProps : focusProps |
| 89 | }; |
| 90 | } |