| 36 | * Focus events on child elements will be ignored. |
| 37 | */ |
| 38 | export function useFocus<Target extends FocusableElement = FocusableElement>( |
| 39 | props: FocusProps<Target> |
| 40 | ): FocusResult<Target> { |
| 41 | let {isDisabled, onFocus: onFocusProp, onBlur: onBlurProp, onFocusChange} = props; |
| 42 | |
| 43 | const onBlur: FocusProps<Target>['onBlur'] = useCallback( |
| 44 | (e: FocusEvent<Target>) => { |
| 45 | if (getEventTarget(e) === e.currentTarget) { |
| 46 | if (onBlurProp) { |
| 47 | onBlurProp(e); |
| 48 | } |
| 49 | |
| 50 | if (onFocusChange) { |
| 51 | onFocusChange(false); |
| 52 | } |
| 53 | |
| 54 | return true; |
| 55 | } |
| 56 | }, |
| 57 | [onBlurProp, onFocusChange] |
| 58 | ); |
| 59 | |
| 60 | const onSyntheticFocus = useSyntheticBlurEvent<Target>(onBlur); |
| 61 | |
| 62 | const onFocus: FocusProps<Target>['onFocus'] = useCallback( |
| 63 | (e: FocusEvent<Target>) => { |
| 64 | // Double check that document.activeElement actually matches e.target in case a previously chained |
| 65 | // focus handler already moved focus somewhere else. |
| 66 | |
| 67 | let eventTarget = getEventTarget(e); |
| 68 | const ownerDocument = getOwnerDocument(eventTarget); |
| 69 | const activeElement = ownerDocument ? getActiveElement(ownerDocument) : getActiveElement(); |
| 70 | if (eventTarget === e.currentTarget && eventTarget === activeElement) { |
| 71 | if (onFocusProp) { |
| 72 | onFocusProp(e); |
| 73 | } |
| 74 | |
| 75 | if (onFocusChange) { |
| 76 | onFocusChange(true); |
| 77 | } |
| 78 | |
| 79 | onSyntheticFocus(e); |
| 80 | } |
| 81 | }, |
| 82 | [onFocusChange, onFocusProp, onSyntheticFocus] |
| 83 | ); |
| 84 | |
| 85 | return { |
| 86 | focusProps: { |
| 87 | onFocus: !isDisabled && (onFocusProp || onFocusChange || onBlurProp) ? onFocus : undefined, |
| 88 | onBlur: !isDisabled && (onBlurProp || onFocusChange) ? onBlur : undefined |
| 89 | } |
| 90 | }; |
| 91 | } |