Some checks are pending
Deploy Volleyball CMS / deploy (push) Waiting to run
88 lines
3.1 KiB
TypeScript
88 lines
3.1 KiB
TypeScript
/*
|
|
* Copyright 2020 Adobe. All rights reserved.
|
|
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License. You may obtain a copy
|
|
* of the License at http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software distributed under
|
|
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
|
|
* OF ANY KIND, either express or implied. See the License for the specific language
|
|
* governing permissions and limitations under the License.
|
|
*/
|
|
|
|
// Portions of the code in this file are based on code from react.
|
|
// Original licensing for the following can be found in the
|
|
// NOTICE file in the root directory of this source tree.
|
|
// See https://github.com/facebook/react/tree/cc7c1aece46a6b69b41958d731e0fd27c94bfc6c/packages/react-interactions
|
|
|
|
import {DOMAttributes, FocusableElement, FocusEvents} from '@react-types/shared';
|
|
import {FocusEvent, useCallback} from 'react';
|
|
import {getActiveElement, getEventTarget, getOwnerDocument} from '@react-aria/utils';
|
|
import {useSyntheticBlurEvent} from './utils';
|
|
|
|
export interface FocusProps<Target = FocusableElement> extends FocusEvents<Target> {
|
|
/** Whether the focus events should be disabled. */
|
|
isDisabled?: boolean
|
|
}
|
|
|
|
export interface FocusResult<Target = FocusableElement> {
|
|
/** Props to spread onto the target element. */
|
|
focusProps: DOMAttributes<Target>
|
|
}
|
|
|
|
/**
|
|
* Handles focus events for the immediate target.
|
|
* Focus events on child elements will be ignored.
|
|
*/
|
|
export function useFocus<Target extends FocusableElement = FocusableElement>(props: FocusProps<Target>): FocusResult<Target> {
|
|
let {
|
|
isDisabled,
|
|
onFocus: onFocusProp,
|
|
onBlur: onBlurProp,
|
|
onFocusChange
|
|
} = props;
|
|
|
|
const onBlur: FocusProps<Target>['onBlur'] = useCallback((e: FocusEvent<Target>) => {
|
|
if (e.target === e.currentTarget) {
|
|
if (onBlurProp) {
|
|
onBlurProp(e);
|
|
}
|
|
|
|
if (onFocusChange) {
|
|
onFocusChange(false);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}, [onBlurProp, onFocusChange]);
|
|
|
|
|
|
const onSyntheticFocus = useSyntheticBlurEvent<Target>(onBlur);
|
|
|
|
const onFocus: FocusProps<Target>['onFocus'] = useCallback((e: FocusEvent<Target>) => {
|
|
// Double check that document.activeElement actually matches e.target in case a previously chained
|
|
// focus handler already moved focus somewhere else.
|
|
|
|
const ownerDocument = getOwnerDocument(e.target);
|
|
const activeElement = ownerDocument ? getActiveElement(ownerDocument) : getActiveElement();
|
|
if (e.target === e.currentTarget && activeElement === getEventTarget(e.nativeEvent)) {
|
|
if (onFocusProp) {
|
|
onFocusProp(e);
|
|
}
|
|
|
|
if (onFocusChange) {
|
|
onFocusChange(true);
|
|
}
|
|
|
|
onSyntheticFocus(e);
|
|
}
|
|
}, [onFocusChange, onFocusProp, onSyntheticFocus]);
|
|
|
|
return {
|
|
focusProps: {
|
|
onFocus: (!isDisabled && (onFocusProp || onFocusChange || onBlurProp)) ? onFocus : undefined,
|
|
onBlur: (!isDisabled && (onBlurProp || onFocusChange)) ? onBlur : undefined
|
|
}
|
|
};
|
|
}
|